Commit c22be15a authored by Huw Davies's avatar Huw Davies Committed by Alexandre Julliard

wineps: Use a file mapping to read the ppd file.

parent 2797cd8e
...@@ -588,21 +588,20 @@ static BOOL set_devmode( HANDLE printer, PSDRV_DEVMODE *dm ) ...@@ -588,21 +588,20 @@ static BOOL set_devmode( HANDLE printer, PSDRV_DEVMODE *dm )
return SetPrinterW( printer, 9, (BYTE *)&info, 0 ); return SetPrinterW( printer, 9, (BYTE *)&info, 0 );
} }
static char *get_ppd_filename( HANDLE printer ) static WCHAR *get_ppd_filename( HANDLE printer )
{ {
DWORD needed; DWORD needed;
DRIVER_INFO_2W *info; DRIVER_INFO_2W *info;
char *unixname; WCHAR *name;
GetPrinterDriverW( printer, NULL, 2, NULL, 0, &needed ); GetPrinterDriverW( printer, NULL, 2, NULL, 0, &needed );
if (GetLastError() != ERROR_INSUFFICIENT_BUFFER) return NULL; if (GetLastError() != ERROR_INSUFFICIENT_BUFFER) return NULL;
info = HeapAlloc( GetProcessHeap(), 0, needed ); info = HeapAlloc( GetProcessHeap(), 0, needed );
if (!info) return NULL; if (!info) return NULL;
GetPrinterDriverW( printer, NULL, 2, (BYTE*)info, needed, &needed ); GetPrinterDriverW( printer, NULL, 2, (BYTE*)info, needed, &needed );
unixname = wine_get_unix_file_name( info->pDataFile ); name = (WCHAR *)info;
HeapFree( GetProcessHeap(), 0, info ); memmove( name, info->pDataFile, (strlenW( info->pDataFile ) + 1) * sizeof(WCHAR) );
return name;
return unixname;
} }
static struct list printer_list = LIST_INIT( printer_list ); static struct list printer_list = LIST_INIT( printer_list );
...@@ -616,7 +615,8 @@ PRINTERINFO *PSDRV_FindPrinterInfo(LPCWSTR name) ...@@ -616,7 +615,8 @@ PRINTERINFO *PSDRV_FindPrinterInfo(LPCWSTR name)
FONTNAME *font; FONTNAME *font;
const AFM *afm; const AFM *afm;
HANDLE hPrinter = 0; HANDLE hPrinter = 0;
char *ppd_filename = NULL, *nameA = NULL; WCHAR *ppd_filename = NULL;
char *nameA = NULL;
BOOL using_default_devmode = FALSE; BOOL using_default_devmode = FALSE;
int len; int len;
...@@ -652,7 +652,7 @@ PRINTERINFO *PSDRV_FindPrinterInfo(LPCWSTR name) ...@@ -652,7 +652,7 @@ PRINTERINFO *PSDRV_FindPrinterInfo(LPCWSTR name)
pi->ppd = PSDRV_ParsePPD( ppd_filename, hPrinter ); pi->ppd = PSDRV_ParsePPD( ppd_filename, hPrinter );
if (!pi->ppd) if (!pi->ppd)
{ {
WARN( "Couldn't parse PPD file '%s'\n", ppd_filename ); WARN( "Couldn't parse PPD file %s\n", debugstr_w(ppd_filename) );
goto fail; goto fail;
} }
......
...@@ -27,6 +27,7 @@ ...@@ -27,6 +27,7 @@
#include <stdio.h> #include <stdio.h>
#include <ctype.h> #include <ctype.h>
#include <locale.h> #include <locale.h>
#include <assert.h>
#include "windef.h" #include "windef.h"
#include "winbase.h" #include "winbase.h"
#include "wine/debug.h" #include "wine/debug.h"
...@@ -42,6 +43,10 @@ char *value; ...@@ -42,6 +43,10 @@ char *value;
char *valtrans; char *valtrans;
} PPDTuple; } PPDTuple;
struct map_context
{
const char *ptr, *pos, *end;
};
/* map of page names in ppd file to Windows paper constants */ /* map of page names in ppd file to Windows paper constants */
...@@ -236,9 +241,10 @@ static char *PSDRV_PPDDecodeHex(char *str) ...@@ -236,9 +241,10 @@ static char *PSDRV_PPDDecodeHex(char *str)
* PSDRV_PPDGetTransValue * PSDRV_PPDGetTransValue
* *
*/ */
static BOOL PSDRV_PPDGetTransValue(char *start, PPDTuple *tuple) static BOOL PSDRV_PPDGetTransValue(const char *start, PPDTuple *tuple)
{ {
char *buf, *end; char *buf;
const char *end;
end = strpbrk(start, "\r\n"); end = strpbrk(start, "\r\n");
if(end == start) return FALSE; if(end == start) return FALSE;
...@@ -251,6 +257,24 @@ static BOOL PSDRV_PPDGetTransValue(char *start, PPDTuple *tuple) ...@@ -251,6 +257,24 @@ static BOOL PSDRV_PPDGetTransValue(char *start, PPDTuple *tuple)
return TRUE; return TRUE;
} }
static BOOL get_line( char *buf, int size, struct map_context *ctx )
{
int i;
if (ctx->pos > ctx->end) return FALSE;
for (i = 0; i < size - 1; i++)
{
if (ctx->pos > ctx->end) break;
buf[i] = *ctx->pos++;
if (buf[i] == '\n')
{
i++;
break;
}
}
buf[i] = '\0';
return TRUE;
}
/*********************************************************************** /***********************************************************************
* *
...@@ -259,38 +283,30 @@ static BOOL PSDRV_PPDGetTransValue(char *start, PPDTuple *tuple) ...@@ -259,38 +283,30 @@ static BOOL PSDRV_PPDGetTransValue(char *start, PPDTuple *tuple)
* Passed string that should be surrounded by `"'s, return string alloced * Passed string that should be surrounded by `"'s, return string alloced
* from process heap. * from process heap.
*/ */
static BOOL PSDRV_PPDGetInvocationValue(FILE *fp, char *pos, PPDTuple *tuple) static BOOL PSDRV_PPDGetInvocationValue(struct map_context *ctx, PPDTuple *tuple)
{ {
char *start, *end, *buf; const char *start;
char line[257]; char *buf, line[257];
int len;
start = pos + 1; assert( *ctx->pos == '"' );
buf = HeapAlloc( PSDRV_Heap, 0, strlen(start) + 1 );
len = 0; ctx->pos++;
do { for (start = ctx->pos; ctx->pos <= ctx->end; ctx->pos++)
end = strchr(start, '"'); if (*ctx->pos == '"') break;
if(end) { if (ctx->pos > ctx->end) return FALSE;
buf = HeapReAlloc( PSDRV_Heap, 0, buf, ctx->pos++;
len + (end - start) + 1 );
memcpy(buf + len, start, end - start); buf = HeapAlloc( PSDRV_Heap, 0, ctx->pos - start );
*(buf + len + (end - start)) = '\0'; memcpy( buf, start, ctx->pos - start - 1 );
buf[ctx->pos - start - 1] = '\0';
tuple->value = buf; tuple->value = buf;
start = strchr(end, '/');
if(start)
return PSDRV_PPDGetTransValue(start + 1, tuple);
return TRUE;
} else {
int sl = strlen(start);
buf = HeapReAlloc( PSDRV_Heap, 0, buf, len + sl + 1 );
strcpy(buf + len, start);
len += sl;
}
} while( fgets((start = line), sizeof(line), fp) );
tuple->value = NULL; if (get_line( line, sizeof(line), ctx ))
HeapFree( PSDRV_Heap, 0, buf ); {
return FALSE; start = strchr( line, '/' );
if (start) return PSDRV_PPDGetTransValue( start + 1, tuple );
}
return TRUE;
} }
...@@ -301,11 +317,11 @@ static BOOL PSDRV_PPDGetInvocationValue(FILE *fp, char *pos, PPDTuple *tuple) ...@@ -301,11 +317,11 @@ static BOOL PSDRV_PPDGetInvocationValue(FILE *fp, char *pos, PPDTuple *tuple)
* Passed string that should be surrounded by `"'s. Expand <xx> as hex * Passed string that should be surrounded by `"'s. Expand <xx> as hex
* return string alloced from process heap. * return string alloced from process heap.
*/ */
static BOOL PSDRV_PPDGetQuotedValue(FILE *fp, char *pos, PPDTuple *tuple) static BOOL PSDRV_PPDGetQuotedValue(struct map_context *ctx, PPDTuple *tuple)
{ {
char *buf; char *buf;
if(!PSDRV_PPDGetInvocationValue(fp, pos, tuple)) if(!PSDRV_PPDGetInvocationValue(ctx, tuple))
return FALSE; return FALSE;
buf = PSDRV_PPDDecodeHex(tuple->value); buf = PSDRV_PPDDecodeHex(tuple->value);
HeapFree(PSDRV_Heap, 0, tuple->value); HeapFree(PSDRV_Heap, 0, tuple->value);
...@@ -358,10 +374,11 @@ static BOOL PSDRV_PPDGetSymbolValue(char *pos, PPDTuple *tuple) ...@@ -358,10 +374,11 @@ static BOOL PSDRV_PPDGetSymbolValue(char *pos, PPDTuple *tuple)
* Gets the next Keyword Option Value tuple from the file. Allocs space off * Gets the next Keyword Option Value tuple from the file. Allocs space off
* the process heap which should be free()ed by the caller if not needed. * the process heap which should be free()ed by the caller if not needed.
*/ */
static BOOL PSDRV_PPDGetNextTuple(FILE *fp, PPDTuple *tuple) static BOOL PSDRV_PPDGetNextTuple(struct map_context *ctx, PPDTuple *tuple)
{ {
char line[257], *opt, *cp, *trans, *endkey; char line[257], *opt, *cp, *trans, *endkey;
BOOL gotoption; BOOL gotoption;
struct map_context save;
start: start:
...@@ -370,7 +387,8 @@ static BOOL PSDRV_PPDGetNextTuple(FILE *fp, PPDTuple *tuple) ...@@ -370,7 +387,8 @@ static BOOL PSDRV_PPDGetNextTuple(FILE *fp, PPDTuple *tuple)
memset(tuple, 0, sizeof(*tuple)); memset(tuple, 0, sizeof(*tuple));
do { do {
if(!fgets(line, sizeof(line), fp)) save = *ctx;
if(!get_line(line, sizeof(line), ctx))
return FALSE; return FALSE;
if(line[0] == '*' && line[1] != '%' && strncmp(line, "*End", 4)) if(line[0] == '*' && line[1] != '%' && strncmp(line, "*End", 4))
break; break;
...@@ -435,11 +453,13 @@ static BOOL PSDRV_PPDGetNextTuple(FILE *fp, PPDTuple *tuple) ...@@ -435,11 +453,13 @@ static BOOL PSDRV_PPDGetNextTuple(FILE *fp, PPDTuple *tuple)
switch(*cp) { switch(*cp) {
case '"': case '"':
/* update the context pos so that it points to the opening quote */
ctx->pos = save.pos + (cp - line);
if( (!gotoption && strncmp(tuple->key, "*?", 2) ) || if( (!gotoption && strncmp(tuple->key, "*?", 2) ) ||
!strncmp(tuple->key, "*JCL", 4)) !strncmp(tuple->key, "*JCL", 4))
PSDRV_PPDGetQuotedValue(fp, cp, tuple); PSDRV_PPDGetQuotedValue(ctx, tuple);
else else
PSDRV_PPDGetInvocationValue(fp, cp, tuple); PSDRV_PPDGetInvocationValue(ctx, tuple);
break; break;
case '^': case '^':
...@@ -622,32 +642,62 @@ static char *get_ppd_override( HANDLE printer, const char *value ) ...@@ -622,32 +642,62 @@ static char *get_ppd_override( HANDLE printer, const char *value )
return data; return data;
} }
static BOOL map_file( const WCHAR *filename, struct map_context *c )
{
HANDLE file, mapping;
LARGE_INTEGER size;
file = CreateFileW( filename, GENERIC_READ, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL );
if (file == INVALID_HANDLE_VALUE) return FALSE;
if (!GetFileSizeEx( file, &size ) || size.u.HighPart)
{
CloseHandle( file );
return FALSE;
}
mapping = CreateFileMappingW( file, NULL, PAGE_READONLY, 0, 0, NULL );
CloseHandle( file );
if (!mapping) return FALSE;
c->pos = c->ptr = MapViewOfFile( mapping, FILE_MAP_READ, 0, 0, 0 );
c->end = c->ptr + size.u.LowPart - 1;
CloseHandle( mapping );
return TRUE;
}
static void unmap_file( struct map_context *c )
{
UnmapViewOfFile( c->ptr );
}
/*********************************************************************** /***********************************************************************
* *
* PSDRV_ParsePPD * PSDRV_ParsePPD
* *
* *
*/ */
PPD *PSDRV_ParsePPD( char *fname, HANDLE printer ) PPD *PSDRV_ParsePPD( const WCHAR *fname, HANDLE printer )
{ {
FILE *fp;
PPD *ppd; PPD *ppd;
PPDTuple tuple; PPDTuple tuple;
char *default_pagesize = NULL, *default_duplex = NULL; char *default_pagesize = NULL, *default_duplex = NULL;
char *def_pagesize_override = NULL, *def_duplex_override = NULL; char *def_pagesize_override = NULL, *def_duplex_override = NULL;
PAGESIZE *page, *page_cursor2; PAGESIZE *page, *page_cursor2;
struct map_context c;
TRACE("file '%s'\n", fname); TRACE("file %s\n", debugstr_w(fname));
if((fp = fopen(fname, "r")) == NULL) { if (!map_file( fname, &c ))
WARN("Couldn't open ppd file '%s'\n", fname); {
WARN("Couldn't open ppd file %s\n", debugstr_w(fname));
return NULL; return NULL;
} }
ppd = HeapAlloc( PSDRV_Heap, HEAP_ZERO_MEMORY, sizeof(*ppd)); ppd = HeapAlloc( PSDRV_Heap, HEAP_ZERO_MEMORY, sizeof(*ppd));
if(!ppd) { if(!ppd) {
ERR("Unable to allocate memory for ppd\n"); ERR("Unable to allocate memory for ppd\n");
fclose(fp); unmap_file( &c );
return NULL; return NULL;
} }
...@@ -671,12 +721,12 @@ PPD *PSDRV_ParsePPD( char *fname, HANDLE printer ) ...@@ -671,12 +721,12 @@ PPD *PSDRV_ParsePPD( char *fname, HANDLE printer )
if (!PSDRV_AddSlot( ppd, NULL, "Automatically Select", NULL, DMBIN_FORMSOURCE )) if (!PSDRV_AddSlot( ppd, NULL, "Automatically Select", NULL, DMBIN_FORMSOURCE ))
{ {
HeapFree (PSDRV_Heap, 0, ppd); HeapFree (PSDRV_Heap, 0, ppd);
fclose(fp); unmap_file( &c );
return NULL; return NULL;
} }
while( PSDRV_PPDGetNextTuple(fp, &tuple)) { while (PSDRV_PPDGetNextTuple( &c, &tuple ))
{
if(!strcmp("*NickName", tuple.key)) { if(!strcmp("*NickName", tuple.key)) {
ppd->NickName = tuple.value; ppd->NickName = tuple.value;
tuple.value = NULL; tuple.value = NULL;
...@@ -1044,6 +1094,6 @@ PPD *PSDRV_ParsePPD( char *fname, HANDLE printer ) ...@@ -1044,6 +1094,6 @@ PPD *PSDRV_ParsePPD( char *fname, HANDLE printer )
debugstr_a(slot->InvocationString)); debugstr_a(slot->InvocationString));
} }
fclose(fp); unmap_file( &c );
return ppd; return ppd;
} }
...@@ -465,7 +465,7 @@ extern BOOL PSDRV_StrokePath( PHYSDEV dev ) DECLSPEC_HIDDEN; ...@@ -465,7 +465,7 @@ extern BOOL PSDRV_StrokePath( PHYSDEV dev ) DECLSPEC_HIDDEN;
extern void PSDRV_MergeDevmodes(PSDRV_DEVMODE *dm1, const PSDRV_DEVMODE *dm2, extern void PSDRV_MergeDevmodes(PSDRV_DEVMODE *dm1, const PSDRV_DEVMODE *dm2,
PRINTERINFO *pi) DECLSPEC_HIDDEN; PRINTERINFO *pi) DECLSPEC_HIDDEN;
extern BOOL PSDRV_GetFontMetrics(void) DECLSPEC_HIDDEN; extern BOOL PSDRV_GetFontMetrics(void) DECLSPEC_HIDDEN;
extern PPD *PSDRV_ParsePPD(char *fname, HANDLE printer) DECLSPEC_HIDDEN; extern PPD *PSDRV_ParsePPD(const WCHAR *fname, HANDLE printer) DECLSPEC_HIDDEN;
extern PRINTERINFO *PSDRV_FindPrinterInfo(LPCWSTR name) DECLSPEC_HIDDEN; extern PRINTERINFO *PSDRV_FindPrinterInfo(LPCWSTR name) DECLSPEC_HIDDEN;
extern const AFM *PSDRV_FindAFMinList(FONTFAMILY *head, LPCSTR name) DECLSPEC_HIDDEN; extern const AFM *PSDRV_FindAFMinList(FONTFAMILY *head, LPCSTR name) DECLSPEC_HIDDEN;
extern BOOL PSDRV_AddAFMtoList(FONTFAMILY **head, const AFM *afm, extern BOOL PSDRV_AddAFMtoList(FONTFAMILY **head, const AFM *afm,
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment