/* Copyright 1996, 1998 The Open Group Permission to use, copy, modify, distribute, and sell this software and its documentation for any purpose is hereby granted without fee, provided that the above copyright notice appear in all copies and that both that copyright notice and this permission notice appear in supporting documentation. The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE OPEN GROUP BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. Except as contained in this notice, the name of The Open Group shall not be used in advertising or otherwise to promote the sale, use or other dealings in this Software without prior written authorization from The Open Group. */ /* * (c) Copyright 1996 Hewlett-Packard Company * (c) Copyright 1996 International Business Machines Corp. * (c) Copyright 1996 Sun Microsystems, Inc. * (c) Copyright 1996 Novell, Inc. * (c) Copyright 1996 Digital Equipment Corp. * (c) Copyright 1996 Fujitsu Limited * (c) Copyright 1996 Hitachi, Ltd. * * Permission is hereby granted, free of charge, to any person obtaining * a copy of this software and associated documentation files (the * "Software"), to deal in the Software without restriction, including * without limitation the rights to use, copy, modify, merge, publish, * distribute, sublicense, and/or sell copies of the Software, and to * permit persons to whom the Software is furnished to do so, subject * to the following conditions: * * The above copyright notice and this permission notice shall be included * in all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL * THE COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, * WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF * OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE * SOFTWARE. * * Except as contained in this notice, the names of the copyright holders * shall not be used in advertising or otherwise to promote the sale, use * or other dealings in this Software without prior written authorization * from said copyright holders. */ /******************************************************************* ** ** ********************************************************* ** * ** * File: PsInit.c ** * ** * Contents: Initialization code of Ps driver for the print server. ** * ** * Created By: Roger Helmendach (Liberty Systems) ** * ** * Copyright: Copyright 1996 The Open Group, Inc. ** * ** ********************************************************* ** ********************************************************************/ #ifdef HAVE_DIX_CONFIG_H #include #endif #include #include #include #include #include #include #include "Ps.h" #include "mi.h" #include "micmap.h" #include "AttrValid.h" #include "fb.h" #include "windowstr.h" #include "DiPrint.h" static void AllocatePsPrivates(ScreenPtr pScreen); static int PsInitContext(XpContextPtr pCon); static int PsDestroyContext(XpContextPtr pCon); int PsScreenPrivateIndex; int PsContextPrivateIndex; int PsPixmapPrivateIndex; int PsWindowPrivateIndex; #ifdef GLXEXT extern void GlxWrapInitVisuals(miInitVisualsProcPtr *); #endif /* GLXEXT */ Bool InitializePsDriver(ndx, pScreen, argc, argv) int ndx; ScreenPtr pScreen; int argc; char **argv; { #if 0 int maxXres, maxYres, maxWidth, maxHeight; int maxRes, maxDim, numBytes; PsScreenPrivPtr pPriv; #endif int nv, /* total number of visuals */ nv_1bit, /* number of 8bit visuals */ nv_8bit, /* number of 8bit visuals */ nv_12bit, /* number of 12bit visuals */ nv_14bit, /* number of 14bit visuals */ nv_16bit, /* number of 16bit visuals */ nv_24bit, /* number of 24bit visuals*/ nv_30bit; /* number of 30bit visuals*/ int nd; /* number of depths */ int defaultVisualIndex = -1; VisualID *vids_1bit, *vids_8bit, *vids_12bit, *vids_14bit, *vids_16bit, *vids_24bit, *vids_30bit; VisualPtr visuals; DepthPtr depths; VisualID defaultVisual; int rootDepth; /* * Register this driver's InitContext function with the print * extension. */ XpRegisterInitFunc(pScreen, "XP-POSTSCRIPT", PsInitContext); /* * Create and fill in the devPrivate for the PS driver. */ AllocatePsPrivates(pScreen); #if 0 pPriv = (PsScreenPrivPtr)pScreen->devPrivates[PsScreenPrivateIndex].ptr; pPriv->resDB = rmdb; #endif pScreen->defColormap = (Colormap) FakeClientID(0); pScreen->blackPixel = 1; pScreen->whitePixel = 0; pScreen->QueryBestSize = (QueryBestSizeProcPtr)PsQueryBestSize; pScreen->SaveScreen = (SaveScreenProcPtr)_XpBoolNoop; pScreen->GetImage = (GetImageProcPtr)_XpVoidNoop; pScreen->GetSpans = (GetSpansProcPtr)_XpVoidNoop; pScreen->CreateWindow = PsCreateWindow; pScreen->DestroyWindow = PsDestroyWindow; pScreen->PositionWindow = PsPositionWindow; pScreen->ChangeWindowAttributes = PsChangeWindowAttributes; pScreen->RealizeWindow = PsMapWindow; pScreen->UnrealizeWindow = PsUnmapWindow; pScreen->PaintWindowBackground = PsPaintWindow; pScreen->PaintWindowBorder = PsPaintWindow; pScreen->CloseScreen = PsCloseScreen; pScreen->CopyWindow = PsCopyWindow; /* XXX Hard routine to write! */ /* * These two are going to be VERY different... */ pScreen->CreatePixmap = PsCreatePixmap; pScreen->DestroyPixmap = PsDestroyPixmap; pScreen->RealizeFont = PsRealizeFont; pScreen->UnrealizeFont = PsUnrealizeFont; pScreen->CreateGC = PsCreateGC; pScreen->CreateColormap = PsCreateColormap; pScreen->DestroyColormap = PsDestroyColormap; pScreen->InstallColormap = PsInstallColormap; pScreen->UninstallColormap = PsUninstallColormap; pScreen->ListInstalledColormaps = PsListInstalledColormaps; pScreen->StoreColors = PsStoreColors; pScreen->ResolveColor = PsResolveColor; /* Will BitmapToRegion make any difference at all? */ pScreen->BitmapToRegion = fbPixmapToRegion; visuals = (VisualPtr) xalloc(16*sizeof(VisualRec)); depths = (DepthPtr) xalloc(16*sizeof(DepthRec)); vids_1bit = (VisualID *)xalloc(16*sizeof(VisualID)); vids_8bit = (VisualID *)xalloc(16*sizeof(VisualID)); vids_12bit = (VisualID *)xalloc(16*sizeof(VisualID)); vids_14bit = (VisualID *)xalloc(16*sizeof(VisualID)); vids_16bit = (VisualID *)xalloc(16*sizeof(VisualID)); vids_24bit = (VisualID *)xalloc(16*sizeof(VisualID)); vids_30bit = (VisualID *)xalloc(16*sizeof(VisualID)); nv = nv_1bit = nv_8bit = nv_12bit = nv_14bit = nv_16bit = nv_24bit = nv_30bit = nd = 0; #ifdef PSOUT_USE_DEEPCOLOR /* gisburn: 30bit TrueColor has been disabled for now since it causes problems * with GLX - see https://bugs.freedesktop.org/show_bug.cgi?id=2868 ("Mesa * seems to be unable to handle 30bit TrueColor visuals") for details... */ #ifdef DISABLED_FOR_NOW /* TrueColor, 30bit, 10bit per R-,G-,B-gun */ visuals[nv].vid = FakeClientID(0); visuals[nv].class = TrueColor; visuals[nv].bitsPerRGBValue = 10; visuals[nv].ColormapEntries = 1024; visuals[nv].nplanes = 30; visuals[nv].redMask = 0X3FF00000; visuals[nv].greenMask = 0X000FFC00; visuals[nv].blueMask = 0X000003FF; visuals[nv].offsetRed = 20; visuals[nv].offsetGreen = 10; visuals[nv].offsetBlue = 0; vids_30bit[nv_30bit] = visuals[nv].vid; nv++; nv_30bit++; #endif /* DISABLED_FOR_NOW */ #endif /* PSOUT_USE_DEEPCOLOR */ /* TrueColor, 24bit */ visuals[nv].vid = FakeClientID(0); visuals[nv].class = TrueColor; visuals[nv].bitsPerRGBValue = 8; visuals[nv].ColormapEntries = 256; visuals[nv].nplanes = 24; visuals[nv].redMask = 0X00FF0000; visuals[nv].greenMask = 0X0000FF00; visuals[nv].blueMask = 0X000000FF; visuals[nv].offsetRed = 16; visuals[nv].offsetGreen = 8; visuals[nv].offsetBlue = 0; vids_24bit[nv_24bit] = visuals[nv].vid; nv++; nv_24bit++; /* TrueColor, 16bit */ visuals[nv].vid = FakeClientID(0); visuals[nv].class = TrueColor; visuals[nv].bitsPerRGBValue = 6; visuals[nv].ColormapEntries = 64; visuals[nv].nplanes = 16; visuals[nv].redMask = 0x0000f800; visuals[nv].greenMask = 0x000007e0; visuals[nv].blueMask = 0x0000001f; visuals[nv].offsetRed = 11; visuals[nv].offsetGreen = 5; visuals[nv].offsetBlue = 0; vids_16bit[nv_16bit] = visuals[nv].vid; nv++; nv_16bit++; #ifdef PSOUT_USE_DEEPCOLOR /* PostScript Level 2 and above, colors can have 12 bits per component * (36 bit for RGB) */ /* PseudoColor, 14bit (15bit won't work as |ColormapEntries==32768| * is too large for a |signed short|... xx@@!!!... ;-( ) */ visuals[nv].vid = FakeClientID(0); visuals[nv].class = PseudoColor; visuals[nv].bitsPerRGBValue = 12; visuals[nv].ColormapEntries = 16384; visuals[nv].nplanes = 14; visuals[nv].redMask = 0x0; visuals[nv].greenMask = 0x0; visuals[nv].blueMask = 0x0; visuals[nv].offsetRed = 0x0; visuals[nv].offsetGreen = 0x0; visuals[nv].offsetBlue = 0x0; vids_14bit[nv_14bit] = visuals[nv].vid; nv++; nv_14bit++; /* PseudoColor, 12bit */ visuals[nv].vid = FakeClientID(0); visuals[nv].class = PseudoColor; visuals[nv].bitsPerRGBValue = 12; visuals[nv].ColormapEntries = 4096; visuals[nv].nplanes = 12; visuals[nv].redMask = 0x0; visuals[nv].greenMask = 0x0; visuals[nv].blueMask = 0x0; visuals[nv].offsetRed = 0x0; visuals[nv].offsetGreen = 0x0; visuals[nv].offsetBlue = 0x0; vids_12bit[nv_12bit] = visuals[nv].vid; defaultVisualIndex = nv; nv++; nv_12bit++; /* GrayScale, 12bit, 12bit per R-,G-,B-gun */ visuals[nv].vid = FakeClientID(0); visuals[nv].class = GrayScale; visuals[nv].bitsPerRGBValue = 12; visuals[nv].ColormapEntries = 4096; visuals[nv].nplanes = 12; visuals[nv].redMask = 0x0; visuals[nv].greenMask = 0x0; visuals[nv].blueMask = 0x0; visuals[nv].offsetRed = 0x0; visuals[nv].offsetGreen = 0x0; visuals[nv].offsetBlue = 0x0; vids_12bit[nv_12bit] = visuals[nv].vid; nv++; nv_12bit++; /* StaticGray, 12bit, 12bit per R-,G-,B-gun */ visuals[nv].vid = FakeClientID(0); visuals[nv].class = StaticGray; visuals[nv].bitsPerRGBValue = 12; visuals[nv].ColormapEntries = 4096; visuals[nv].nplanes = 12; visuals[nv].redMask = 0x0; visuals[nv].greenMask = 0x0; visuals[nv].blueMask = 0x0; visuals[nv].offsetRed = 0x0; visuals[nv].offsetGreen = 0x0; visuals[nv].offsetBlue = 0x0; vids_12bit[nv_12bit] = visuals[nv].vid; nv++; nv_12bit++; #endif /* PSOUT_USE_DEEPCOLOR */ /* PseudoColor, 8bit */ visuals[nv].vid = FakeClientID(0); visuals[nv].class = PseudoColor; visuals[nv].bitsPerRGBValue = 8; visuals[nv].ColormapEntries = 256; visuals[nv].nplanes = 8; visuals[nv].redMask = 0x0; visuals[nv].greenMask = 0x0; visuals[nv].blueMask = 0x0; visuals[nv].offsetRed = 0x0; visuals[nv].offsetGreen = 0x0; visuals[nv].offsetBlue = 0x0; vids_8bit[nv_8bit] = visuals[nv].vid; #ifndef PSOUT_USE_DEEPCOLOR defaultVisualIndex = nv; #endif /* !PSOUT_USE_DEEPCOLOR */ nv++; nv_8bit++; /* GrayScale, 8bit */ visuals[nv].vid = FakeClientID(0); visuals[nv].class = GrayScale; visuals[nv].bitsPerRGBValue = 8; visuals[nv].ColormapEntries = 256; visuals[nv].nplanes = 8; visuals[nv].redMask = 0x0; visuals[nv].greenMask = 0x0; visuals[nv].blueMask = 0x0; visuals[nv].offsetRed = 0x0; visuals[nv].offsetGreen = 0x0; visuals[nv].offsetBlue = 0x0; vids_8bit[nv_8bit] = visuals[nv].vid; nv++; nv_8bit++; /* StaticGray, 8bit */ visuals[nv].vid = FakeClientID(0); visuals[nv].class = StaticGray; visuals[nv].bitsPerRGBValue = 8; visuals[nv].ColormapEntries = 256; visuals[nv].nplanes = 8; visuals[nv].redMask = 0x0; visuals[nv].greenMask = 0x0; visuals[nv].blueMask = 0x0; visuals[nv].offsetRed = 0x0; visuals[nv].offsetGreen = 0x0; visuals[nv].offsetBlue = 0x0; vids_8bit[nv_8bit] = visuals[nv].vid; nv++; nv_8bit++; /* StaticGray, 1bit */ visuals[nv].vid = FakeClientID(0); visuals[nv].class = StaticGray; visuals[nv].bitsPerRGBValue = 1; visuals[nv].ColormapEntries = 2; visuals[nv].nplanes = 1; visuals[nv].redMask = 0x0; visuals[nv].greenMask = 0x0; visuals[nv].blueMask = 0x0; visuals[nv].offsetRed = 0x0; visuals[nv].offsetGreen = 0x0; visuals[nv].offsetBlue = 0x0; vids_1bit[nv_1bit] = visuals[nv].vid; nv++; nv_1bit++; if( nv_30bit > 0 ) { depths[nd].depth = 30; depths[nd].numVids = nv_30bit; depths[nd].vids = vids_30bit; nd++; } if( nv_24bit > 0 ) { depths[nd].depth = 24; depths[nd].numVids = nv_24bit; depths[nd].vids = vids_24bit; nd++; } if( nv_16bit > 0 ) { depths[nd].depth = 16; depths[nd].numVids = nv_16bit; depths[nd].vids = vids_16bit; nd++; } if( nv_14bit > 0 ) { depths[nd].depth = 14; depths[nd].numVids = nv_14bit; depths[nd].vids = vids_14bit; nd++; } if( nv_12bit > 0 ) { depths[nd].depth = 12; depths[nd].numVids = nv_12bit; depths[nd].vids = vids_12bit; nd++; } if( nv_8bit > 0 ) { depths[nd].depth = 8; depths[nd].numVids = nv_8bit; depths[nd].vids = vids_8bit; nd++; } if( nv_1bit > 0 ) { depths[nd].depth = 1; depths[nd].numVids = nv_1bit; depths[nd].vids = vids_1bit; nd++; } /* Defaul visual is 12bit PseudoColor */ defaultVisual = visuals[defaultVisualIndex].vid; rootDepth = visuals[defaultVisualIndex].nplanes; #ifdef GLXEXT { miInitVisualsProcPtr proc = NULL; GlxWrapInitVisuals(&proc); /* GlxInitVisuals ignores the last three arguments. */ proc(&visuals, &depths, &nv, &nd, &rootDepth, &defaultVisual, 0, 0, 0); } #endif /* GLXEXT */ miScreenInit(pScreen, (pointer)0, pScreen->width, pScreen->height, (int) (pScreen->width / (pScreen->mmWidth / 25.40)), (int) (pScreen->height / (pScreen->mmHeight / 25.40)), 0, rootDepth, nd, depths, defaultVisual, nv, visuals); if( miCreateDefColormap(pScreen)==FALSE ) return FALSE; /*scalingScreenInit(pScreen);*/ return TRUE; } static void AllocatePsPrivates(ScreenPtr pScreen) { static unsigned long PsGeneration = 0; if((unsigned long)PsGeneration != serverGeneration) { PsScreenPrivateIndex = AllocateScreenPrivateIndex(); PsWindowPrivateIndex = AllocateWindowPrivateIndex(); AllocateWindowPrivate(pScreen, PsWindowPrivateIndex, sizeof(PsWindowPrivRec)); PsContextPrivateIndex = XpAllocateContextPrivateIndex(); XpAllocateContextPrivate(PsContextPrivateIndex, sizeof(PsContextPrivRec)); PsPixmapPrivateIndex = AllocatePixmapPrivateIndex(); AllocatePixmapPrivate(pScreen, PsPixmapPrivateIndex, sizeof(PsPixmapPrivRec)); PsGeneration = serverGeneration; } pScreen->devPrivates[PsScreenPrivateIndex].ptr = (pointer)xalloc(sizeof(PsScreenPrivRec)); } /* * PsInitContext * * Establish the appropriate values for a PrintContext used with the PS * driver. */ static char DOC_ATT_SUPP[]="document-attributes-supported"; static char DOC_ATT_VAL[]="document-format xp-listfonts-modes"; static char JOB_ATT_SUPP[]="job-attributes-supported"; static char JOB_ATT_VAL[]=""; static char PAGE_ATT_SUPP[]="xp-page-attributes-supported"; static char PAGE_ATT_VAL[]="content-orientation default-printer-resolution \ default-input-tray default-medium plex xp-listfonts-modes"; static int PsInitContext(pCon) XpContextPtr pCon; { XpDriverFuncsPtr pFuncs; PsContextPrivPtr pConPriv; char *server, *attrStr; /* * Initialize the attribute store for this printer. */ XpInitAttributes(pCon); /* * Initialize the function pointers */ pFuncs = &(pCon->funcs); pFuncs->StartJob = PsStartJob; pFuncs->EndJob = PsEndJob; pFuncs->StartDoc = PsStartDoc; pFuncs->EndDoc = PsEndDoc; pFuncs->StartPage = PsStartPage; pFuncs->EndPage = PsEndPage; pFuncs->PutDocumentData = PsDocumentData; pFuncs->GetDocumentData = PsGetDocumentData; pFuncs->GetAttributes = PsGetAttributes; pFuncs->SetAttributes = PsSetAttributes; pFuncs->AugmentAttributes = PsAugmentAttributes; pFuncs->GetOneAttribute = PsGetOneAttribute; pFuncs->DestroyContext = PsDestroyContext; pFuncs->GetMediumDimensions = PsGetMediumDimensions; pFuncs->GetReproducibleArea = PsGetReproducibleArea; pFuncs->SetImageResolution = PsSetImageResolution; /* * Set up the context privates */ pConPriv = (PsContextPrivPtr)pCon->devPrivates[PsContextPrivateIndex].ptr; memset(pConPriv, 0, sizeof(PsContextPrivRec)); pConPriv->jobFileName = (char *)NULL; pConPriv->pJobFile = (FILE *)NULL; pConPriv->dash = (unsigned char *)NULL; pConPriv->validGC = 0; pConPriv->getDocClient = (ClientPtr)NULL; pConPriv->getDocBufSize = 0; pConPriv->pPsOut = NULL; pConPriv->fontInfoRecords = NULL; pConPriv->fontTypeInfoRecords = NULL; /* * document-attributes-supported */ server = XpGetOneAttribute( pCon, XPServerAttr, DOC_ATT_SUPP ); if ((attrStr = (char *) xalloc(strlen(server) + strlen(DOC_ATT_SUPP) + strlen(DOC_ATT_VAL) + strlen(PAGE_ATT_VAL) + 8)) == NULL) { return BadAlloc; } sprintf(attrStr, "*%s:\t%s %s %s", DOC_ATT_SUPP, server, DOC_ATT_VAL, PAGE_ATT_VAL); XpAugmentAttributes( pCon, XPPrinterAttr, attrStr); xfree(attrStr); /* * job-attributes-supported */ server = XpGetOneAttribute( pCon, XPServerAttr, JOB_ATT_SUPP ); if ((attrStr = (char *) xalloc(strlen(server) + strlen(JOB_ATT_SUPP) + strlen(JOB_ATT_VAL) + 8)) == NULL) { return BadAlloc; } sprintf(attrStr, "*%s:\t%s %s", JOB_ATT_SUPP, server, JOB_ATT_VAL); XpAugmentAttributes(pCon, XPPrinterAttr, attrStr); xfree(attrStr); /* * xp-page-attributes-supported */ server = XpGetOneAttribute( pCon, XPServerAttr, PAGE_ATT_SUPP ); if ((attrStr = (char *) xalloc(strlen(server) + strlen(PAGE_ATT_SUPP) + strlen(PAGE_ATT_VAL) + 8)) == NULL) { return BadAlloc; } sprintf(attrStr, "*%s:\t%s %s", PAGE_ATT_SUPP, server, PAGE_ATT_VAL); XpAugmentAttributes(pCon, XPPrinterAttr, attrStr); xfree(attrStr); /* * Validate the attribute pools */ XpValidateAttributePool(pCon, XPPrinterAttr, &PsValidatePoolsRec); XpValidateAttributePool(pCon, XPDocAttr, &PsValidatePoolsRec); XpValidateAttributePool(pCon, XPJobAttr, &PsValidatePoolsRec); XpValidateAttributePool(pCon, XPPageAttr, &PsValidatePoolsRec); return Success; } static Bool PsDestroyContext(pCon) XpContextPtr pCon; { PsContextPrivPtr pConPriv = (PsContextPrivPtr)pCon->devPrivates[PsContextPrivateIndex].ptr; if( pConPriv->pJobFile!=(FILE *)NULL ) { fclose(pConPriv->pJobFile); pConPriv->pJobFile = NULL; } if( pConPriv->jobFileName!=(char *)NULL ) { unlink(pConPriv->jobFileName); xfree(pConPriv->jobFileName); pConPriv->jobFileName = (char *)NULL; } PsFreeFontInfoRecords(pConPriv); /* Reset context to make sure we do not use any stale/invalid/obsolete data */ memset(pConPriv, 0, sizeof(PsContextPrivRec)); /*### free up visuals/depths ###*/ return Success; } XpContextPtr PsGetContextFromWindow(win) WindowPtr win; { PsWindowPrivPtr pPriv; while( win ) { pPriv = (PsWindowPrivPtr)win->devPrivates[PsWindowPrivateIndex].ptr; if( pPriv->validContext ) return pPriv->context; win = win->parent; } return NULL; }