cursor.c revision 05b261ec
105b261ecSmrg/*********************************************************** 205b261ecSmrg 305b261ecSmrgCopyright 1987, 1998 The Open Group 405b261ecSmrg 505b261ecSmrgPermission to use, copy, modify, distribute, and sell this software and its 605b261ecSmrgdocumentation for any purpose is hereby granted without fee, provided that 705b261ecSmrgthe above copyright notice appear in all copies and that both that 805b261ecSmrgcopyright notice and this permission notice appear in supporting 905b261ecSmrgdocumentation. 1005b261ecSmrg 1105b261ecSmrgThe above copyright notice and this permission notice shall be included in 1205b261ecSmrgall copies or substantial portions of the Software. 1305b261ecSmrg 1405b261ecSmrgTHE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 1505b261ecSmrgIMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 1605b261ecSmrgFITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 1705b261ecSmrgOPEN GROUP BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN 1805b261ecSmrgAN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN 1905b261ecSmrgCONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. 2005b261ecSmrg 2105b261ecSmrgExcept as contained in this notice, the name of The Open Group shall not be 2205b261ecSmrgused in advertising or otherwise to promote the sale, use or other dealings 2305b261ecSmrgin this Software without prior written authorization from The Open Group. 2405b261ecSmrg 2505b261ecSmrg 2605b261ecSmrgCopyright 1987 by Digital Equipment Corporation, Maynard, Massachusetts. 2705b261ecSmrg 2805b261ecSmrg All Rights Reserved 2905b261ecSmrg 3005b261ecSmrgPermission to use, copy, modify, and distribute this software and its 3105b261ecSmrgdocumentation for any purpose and without fee is hereby granted, 3205b261ecSmrgprovided that the above copyright notice appear in all copies and that 3305b261ecSmrgboth that copyright notice and this permission notice appear in 3405b261ecSmrgsupporting documentation, and that the name of Digital not be 3505b261ecSmrgused in advertising or publicity pertaining to distribution of the 3605b261ecSmrgsoftware without specific, written prior permission. 3705b261ecSmrg 3805b261ecSmrgDIGITAL DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING 3905b261ecSmrgALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL 4005b261ecSmrgDIGITAL BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR 4105b261ecSmrgANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, 4205b261ecSmrgWHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, 4305b261ecSmrgARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS 4405b261ecSmrgSOFTWARE. 4505b261ecSmrg 4605b261ecSmrg******************************************************************/ 4705b261ecSmrg 4805b261ecSmrg 4905b261ecSmrg 5005b261ecSmrg#ifdef HAVE_DIX_CONFIG_H 5105b261ecSmrg#include <dix-config.h> 5205b261ecSmrg#endif 5305b261ecSmrg 5405b261ecSmrg#include <X11/X.h> 5505b261ecSmrg#include <X11/Xmd.h> 5605b261ecSmrg#include "servermd.h" 5705b261ecSmrg#include "scrnintstr.h" 5805b261ecSmrg#include "dixstruct.h" 5905b261ecSmrg#include "cursorstr.h" 6005b261ecSmrg#include "dixfontstr.h" 6105b261ecSmrg#include "opaque.h" 6205b261ecSmrg 6305b261ecSmrgtypedef struct _GlyphShare { 6405b261ecSmrg FontPtr font; 6505b261ecSmrg unsigned short sourceChar; 6605b261ecSmrg unsigned short maskChar; 6705b261ecSmrg CursorBitsPtr bits; 6805b261ecSmrg struct _GlyphShare *next; 6905b261ecSmrg} GlyphShare, *GlyphSharePtr; 7005b261ecSmrg 7105b261ecSmrgstatic GlyphSharePtr sharedGlyphs = (GlyphSharePtr)NULL; 7205b261ecSmrg 7305b261ecSmrg#ifdef XFIXES 7405b261ecSmrgstatic CARD32 cursorSerial; 7505b261ecSmrg#endif 7605b261ecSmrg 7705b261ecSmrgstatic void 7805b261ecSmrgFreeCursorBits(CursorBitsPtr bits) 7905b261ecSmrg{ 8005b261ecSmrg if (--bits->refcnt > 0) 8105b261ecSmrg return; 8205b261ecSmrg xfree(bits->source); 8305b261ecSmrg xfree(bits->mask); 8405b261ecSmrg#ifdef ARGB_CURSOR 8505b261ecSmrg xfree(bits->argb); 8605b261ecSmrg#endif 8705b261ecSmrg if (bits->refcnt == 0) 8805b261ecSmrg { 8905b261ecSmrg GlyphSharePtr *prev, this; 9005b261ecSmrg 9105b261ecSmrg for (prev = &sharedGlyphs; 9205b261ecSmrg (this = *prev) && (this->bits != bits); 9305b261ecSmrg prev = &this->next) 9405b261ecSmrg ; 9505b261ecSmrg if (this) 9605b261ecSmrg { 9705b261ecSmrg *prev = this->next; 9805b261ecSmrg CloseFont(this->font, (Font)0); 9905b261ecSmrg xfree(this); 10005b261ecSmrg } 10105b261ecSmrg xfree(bits); 10205b261ecSmrg } 10305b261ecSmrg} 10405b261ecSmrg 10505b261ecSmrg/** 10605b261ecSmrg * To be called indirectly by DeleteResource; must use exactly two args. 10705b261ecSmrg * 10805b261ecSmrg * \param value must conform to DeleteType 10905b261ecSmrg */ 11005b261ecSmrg_X_EXPORT int 11105b261ecSmrgFreeCursor(pointer value, XID cid) 11205b261ecSmrg{ 11305b261ecSmrg int nscr; 11405b261ecSmrg CursorPtr pCurs = (CursorPtr)value; 11505b261ecSmrg 11605b261ecSmrg ScreenPtr pscr; 11705b261ecSmrg 11805b261ecSmrg if ( --pCurs->refcnt > 0) 11905b261ecSmrg return(Success); 12005b261ecSmrg 12105b261ecSmrg for (nscr = 0; nscr < screenInfo.numScreens; nscr++) 12205b261ecSmrg { 12305b261ecSmrg pscr = screenInfo.screens[nscr]; 12405b261ecSmrg (void)( *pscr->UnrealizeCursor)( pscr, pCurs); 12505b261ecSmrg } 12605b261ecSmrg FreeCursorBits(pCurs->bits); 12705b261ecSmrg xfree( pCurs); 12805b261ecSmrg return(Success); 12905b261ecSmrg} 13005b261ecSmrg 13105b261ecSmrg 13205b261ecSmrg/* 13305b261ecSmrg * We check for empty cursors so that we won't have to display them 13405b261ecSmrg */ 13505b261ecSmrgstatic void 13605b261ecSmrgCheckForEmptyMask(CursorBitsPtr bits) 13705b261ecSmrg{ 13805b261ecSmrg unsigned char *msk = bits->mask; 13905b261ecSmrg int n = BitmapBytePad(bits->width) * bits->height; 14005b261ecSmrg 14105b261ecSmrg bits->emptyMask = FALSE; 14205b261ecSmrg while(n--) 14305b261ecSmrg if(*(msk++) != 0) return; 14405b261ecSmrg#ifdef ARGB_CURSOR 14505b261ecSmrg if (bits->argb) 14605b261ecSmrg { 14705b261ecSmrg CARD32 *argb = bits->argb; 14805b261ecSmrg int n = bits->width * bits->height; 14905b261ecSmrg while (n--) 15005b261ecSmrg if (*argb++ & 0xff000000) return; 15105b261ecSmrg } 15205b261ecSmrg#endif 15305b261ecSmrg bits->emptyMask = TRUE; 15405b261ecSmrg} 15505b261ecSmrg 15605b261ecSmrg/** 15705b261ecSmrg * does nothing about the resource table, just creates the data structure. 15805b261ecSmrg * does not copy the src and mask bits 15905b261ecSmrg * 16005b261ecSmrg * \param psrcbits server-defined padding 16105b261ecSmrg * \param pmaskbits server-defined padding 16205b261ecSmrg * \param argb no padding 16305b261ecSmrg */ 16405b261ecSmrgCursorPtr 16505b261ecSmrgAllocCursorARGB(unsigned char *psrcbits, unsigned char *pmaskbits, CARD32 *argb, 16605b261ecSmrg CursorMetricPtr cm, 16705b261ecSmrg unsigned foreRed, unsigned foreGreen, unsigned foreBlue, 16805b261ecSmrg unsigned backRed, unsigned backGreen, unsigned backBlue) 16905b261ecSmrg{ 17005b261ecSmrg CursorBitsPtr bits; 17105b261ecSmrg CursorPtr pCurs; 17205b261ecSmrg int nscr; 17305b261ecSmrg ScreenPtr pscr; 17405b261ecSmrg 17505b261ecSmrg pCurs = (CursorPtr)xalloc(sizeof(CursorRec) + sizeof(CursorBits)); 17605b261ecSmrg if (!pCurs) 17705b261ecSmrg { 17805b261ecSmrg xfree(psrcbits); 17905b261ecSmrg xfree(pmaskbits); 18005b261ecSmrg return (CursorPtr)NULL; 18105b261ecSmrg } 18205b261ecSmrg bits = (CursorBitsPtr)((char *)pCurs + sizeof(CursorRec)); 18305b261ecSmrg bits->source = psrcbits; 18405b261ecSmrg bits->mask = pmaskbits; 18505b261ecSmrg#ifdef ARGB_CURSOR 18605b261ecSmrg bits->argb = argb; 18705b261ecSmrg#endif 18805b261ecSmrg bits->width = cm->width; 18905b261ecSmrg bits->height = cm->height; 19005b261ecSmrg bits->xhot = cm->xhot; 19105b261ecSmrg bits->yhot = cm->yhot; 19205b261ecSmrg bits->refcnt = -1; 19305b261ecSmrg CheckForEmptyMask(bits); 19405b261ecSmrg 19505b261ecSmrg pCurs->bits = bits; 19605b261ecSmrg pCurs->refcnt = 1; 19705b261ecSmrg#ifdef XFIXES 19805b261ecSmrg pCurs->serialNumber = ++cursorSerial; 19905b261ecSmrg pCurs->name = None; 20005b261ecSmrg#endif 20105b261ecSmrg 20205b261ecSmrg pCurs->foreRed = foreRed; 20305b261ecSmrg pCurs->foreGreen = foreGreen; 20405b261ecSmrg pCurs->foreBlue = foreBlue; 20505b261ecSmrg 20605b261ecSmrg pCurs->backRed = backRed; 20705b261ecSmrg pCurs->backGreen = backGreen; 20805b261ecSmrg pCurs->backBlue = backBlue; 20905b261ecSmrg 21005b261ecSmrg /* 21105b261ecSmrg * realize the cursor for every screen 21205b261ecSmrg */ 21305b261ecSmrg for (nscr = 0; nscr < screenInfo.numScreens; nscr++) 21405b261ecSmrg { 21505b261ecSmrg pscr = screenInfo.screens[nscr]; 21605b261ecSmrg if (!( *pscr->RealizeCursor)( pscr, pCurs)) 21705b261ecSmrg { 21805b261ecSmrg while (--nscr >= 0) 21905b261ecSmrg { 22005b261ecSmrg pscr = screenInfo.screens[nscr]; 22105b261ecSmrg ( *pscr->UnrealizeCursor)( pscr, pCurs); 22205b261ecSmrg } 22305b261ecSmrg FreeCursorBits(bits); 22405b261ecSmrg xfree(pCurs); 22505b261ecSmrg return (CursorPtr)NULL; 22605b261ecSmrg } 22705b261ecSmrg } 22805b261ecSmrg return pCurs; 22905b261ecSmrg} 23005b261ecSmrg 23105b261ecSmrg/** 23205b261ecSmrg * 23305b261ecSmrg * \param psrcbits server-defined padding 23405b261ecSmrg * \param pmaskbits server-defined padding 23505b261ecSmrg */ 23605b261ecSmrgCursorPtr 23705b261ecSmrgAllocCursor(unsigned char *psrcbits, unsigned char *pmaskbits, 23805b261ecSmrg CursorMetricPtr cm, 23905b261ecSmrg unsigned foreRed, unsigned foreGreen, unsigned foreBlue, 24005b261ecSmrg unsigned backRed, unsigned backGreen, unsigned backBlue) 24105b261ecSmrg{ 24205b261ecSmrg return AllocCursorARGB (psrcbits, pmaskbits, (CARD32 *) 0, cm, 24305b261ecSmrg foreRed, foreGreen, foreBlue, 24405b261ecSmrg backRed, backGreen, backBlue); 24505b261ecSmrg} 24605b261ecSmrg 24705b261ecSmrgint 24805b261ecSmrgAllocGlyphCursor(Font source, unsigned sourceChar, Font mask, unsigned maskChar, 24905b261ecSmrg unsigned foreRed, unsigned foreGreen, unsigned foreBlue, 25005b261ecSmrg unsigned backRed, unsigned backGreen, unsigned backBlue, 25105b261ecSmrg CursorPtr *ppCurs, ClientPtr client) 25205b261ecSmrg{ 25305b261ecSmrg FontPtr sourcefont, maskfont; 25405b261ecSmrg unsigned char *srcbits; 25505b261ecSmrg unsigned char *mskbits; 25605b261ecSmrg CursorMetricRec cm; 25705b261ecSmrg int res; 25805b261ecSmrg CursorBitsPtr bits; 25905b261ecSmrg CursorPtr pCurs; 26005b261ecSmrg int nscr; 26105b261ecSmrg ScreenPtr pscr; 26205b261ecSmrg GlyphSharePtr pShare; 26305b261ecSmrg 26405b261ecSmrg sourcefont = (FontPtr) SecurityLookupIDByType(client, source, RT_FONT, 26505b261ecSmrg DixReadAccess); 26605b261ecSmrg maskfont = (FontPtr) SecurityLookupIDByType(client, mask, RT_FONT, 26705b261ecSmrg DixReadAccess); 26805b261ecSmrg 26905b261ecSmrg if (!sourcefont) 27005b261ecSmrg { 27105b261ecSmrg client->errorValue = source; 27205b261ecSmrg return(BadFont); 27305b261ecSmrg } 27405b261ecSmrg if (!maskfont && (mask != None)) 27505b261ecSmrg { 27605b261ecSmrg client->errorValue = mask; 27705b261ecSmrg return(BadFont); 27805b261ecSmrg } 27905b261ecSmrg if (sourcefont != maskfont) 28005b261ecSmrg pShare = (GlyphSharePtr)NULL; 28105b261ecSmrg else 28205b261ecSmrg { 28305b261ecSmrg for (pShare = sharedGlyphs; 28405b261ecSmrg pShare && 28505b261ecSmrg ((pShare->font != sourcefont) || 28605b261ecSmrg (pShare->sourceChar != sourceChar) || 28705b261ecSmrg (pShare->maskChar != maskChar)); 28805b261ecSmrg pShare = pShare->next) 28905b261ecSmrg ; 29005b261ecSmrg } 29105b261ecSmrg if (pShare) 29205b261ecSmrg { 29305b261ecSmrg pCurs = (CursorPtr)xalloc(sizeof(CursorRec)); 29405b261ecSmrg if (!pCurs) 29505b261ecSmrg return BadAlloc; 29605b261ecSmrg bits = pShare->bits; 29705b261ecSmrg bits->refcnt++; 29805b261ecSmrg } 29905b261ecSmrg else 30005b261ecSmrg { 30105b261ecSmrg if (!CursorMetricsFromGlyph(sourcefont, sourceChar, &cm)) 30205b261ecSmrg { 30305b261ecSmrg client->errorValue = sourceChar; 30405b261ecSmrg return BadValue; 30505b261ecSmrg } 30605b261ecSmrg if (!maskfont) 30705b261ecSmrg { 30805b261ecSmrg long n; 30905b261ecSmrg unsigned char *mskptr; 31005b261ecSmrg 31105b261ecSmrg n = BitmapBytePad(cm.width)*(long)cm.height; 31205b261ecSmrg mskptr = mskbits = (unsigned char *)xalloc(n); 31305b261ecSmrg if (!mskptr) 31405b261ecSmrg return BadAlloc; 31505b261ecSmrg while (--n >= 0) 31605b261ecSmrg *mskptr++ = ~0; 31705b261ecSmrg } 31805b261ecSmrg else 31905b261ecSmrg { 32005b261ecSmrg if (!CursorMetricsFromGlyph(maskfont, maskChar, &cm)) 32105b261ecSmrg { 32205b261ecSmrg client->errorValue = maskChar; 32305b261ecSmrg return BadValue; 32405b261ecSmrg } 32505b261ecSmrg if ((res = ServerBitsFromGlyph(maskfont, maskChar, &cm, &mskbits)) != 0) 32605b261ecSmrg return res; 32705b261ecSmrg } 32805b261ecSmrg if ((res = ServerBitsFromGlyph(sourcefont, sourceChar, &cm, &srcbits)) != 0) 32905b261ecSmrg { 33005b261ecSmrg xfree(mskbits); 33105b261ecSmrg return res; 33205b261ecSmrg } 33305b261ecSmrg if (sourcefont != maskfont) 33405b261ecSmrg { 33505b261ecSmrg pCurs = (CursorPtr)xalloc(sizeof(CursorRec) + sizeof(CursorBits)); 33605b261ecSmrg if (pCurs) 33705b261ecSmrg bits = (CursorBitsPtr)((char *)pCurs + sizeof(CursorRec)); 33805b261ecSmrg else 33905b261ecSmrg bits = (CursorBitsPtr)NULL; 34005b261ecSmrg } 34105b261ecSmrg else 34205b261ecSmrg { 34305b261ecSmrg pCurs = (CursorPtr)xalloc(sizeof(CursorRec)); 34405b261ecSmrg if (pCurs) 34505b261ecSmrg bits = (CursorBitsPtr)xalloc(sizeof(CursorBits)); 34605b261ecSmrg else 34705b261ecSmrg bits = (CursorBitsPtr)NULL; 34805b261ecSmrg } 34905b261ecSmrg if (!bits) 35005b261ecSmrg { 35105b261ecSmrg xfree(pCurs); 35205b261ecSmrg xfree(mskbits); 35305b261ecSmrg xfree(srcbits); 35405b261ecSmrg return BadAlloc; 35505b261ecSmrg } 35605b261ecSmrg bits->source = srcbits; 35705b261ecSmrg bits->mask = mskbits; 35805b261ecSmrg#ifdef ARGB_CURSOR 35905b261ecSmrg bits->argb = 0; 36005b261ecSmrg#endif 36105b261ecSmrg bits->width = cm.width; 36205b261ecSmrg bits->height = cm.height; 36305b261ecSmrg bits->xhot = cm.xhot; 36405b261ecSmrg bits->yhot = cm.yhot; 36505b261ecSmrg if (sourcefont != maskfont) 36605b261ecSmrg bits->refcnt = -1; 36705b261ecSmrg else 36805b261ecSmrg { 36905b261ecSmrg bits->refcnt = 1; 37005b261ecSmrg pShare = (GlyphSharePtr)xalloc(sizeof(GlyphShare)); 37105b261ecSmrg if (!pShare) 37205b261ecSmrg { 37305b261ecSmrg FreeCursorBits(bits); 37405b261ecSmrg return BadAlloc; 37505b261ecSmrg } 37605b261ecSmrg pShare->font = sourcefont; 37705b261ecSmrg sourcefont->refcnt++; 37805b261ecSmrg pShare->sourceChar = sourceChar; 37905b261ecSmrg pShare->maskChar = maskChar; 38005b261ecSmrg pShare->bits = bits; 38105b261ecSmrg pShare->next = sharedGlyphs; 38205b261ecSmrg sharedGlyphs = pShare; 38305b261ecSmrg } 38405b261ecSmrg } 38505b261ecSmrg CheckForEmptyMask(bits); 38605b261ecSmrg pCurs->bits = bits; 38705b261ecSmrg pCurs->refcnt = 1; 38805b261ecSmrg#ifdef XFIXES 38905b261ecSmrg pCurs->serialNumber = ++cursorSerial; 39005b261ecSmrg pCurs->name = None; 39105b261ecSmrg#endif 39205b261ecSmrg 39305b261ecSmrg pCurs->foreRed = foreRed; 39405b261ecSmrg pCurs->foreGreen = foreGreen; 39505b261ecSmrg pCurs->foreBlue = foreBlue; 39605b261ecSmrg 39705b261ecSmrg pCurs->backRed = backRed; 39805b261ecSmrg pCurs->backGreen = backGreen; 39905b261ecSmrg pCurs->backBlue = backBlue; 40005b261ecSmrg 40105b261ecSmrg /* 40205b261ecSmrg * realize the cursor for every screen 40305b261ecSmrg */ 40405b261ecSmrg for (nscr = 0; nscr < screenInfo.numScreens; nscr++) 40505b261ecSmrg { 40605b261ecSmrg pscr = screenInfo.screens[nscr]; 40705b261ecSmrg if (!( *pscr->RealizeCursor)( pscr, pCurs)) 40805b261ecSmrg { 40905b261ecSmrg while (--nscr >= 0) 41005b261ecSmrg { 41105b261ecSmrg pscr = screenInfo.screens[nscr]; 41205b261ecSmrg ( *pscr->UnrealizeCursor)( pscr, pCurs); 41305b261ecSmrg } 41405b261ecSmrg FreeCursorBits(pCurs->bits); 41505b261ecSmrg xfree(pCurs); 41605b261ecSmrg return BadAlloc; 41705b261ecSmrg } 41805b261ecSmrg } 41905b261ecSmrg *ppCurs = pCurs; 42005b261ecSmrg return Success; 42105b261ecSmrg} 42205b261ecSmrg 42305b261ecSmrg/** CreateRootCursor 42405b261ecSmrg * 42505b261ecSmrg * look up the name of a font 42605b261ecSmrg * open the font 42705b261ecSmrg * add the font to the resource table 42805b261ecSmrg * make a cursor from the glyphs 42905b261ecSmrg * add the cursor to the resource table 43005b261ecSmrg *************************************************************/ 43105b261ecSmrg 43205b261ecSmrgCursorPtr 43305b261ecSmrgCreateRootCursor(char *unused1, unsigned int unused2) 43405b261ecSmrg{ 43505b261ecSmrg CursorPtr curs; 43605b261ecSmrg#ifdef NULL_ROOT_CURSOR 43705b261ecSmrg CursorMetricRec cm; 43805b261ecSmrg#else 43905b261ecSmrg FontPtr cursorfont; 44005b261ecSmrg int err; 44105b261ecSmrg XID fontID; 44205b261ecSmrg#endif 44305b261ecSmrg 44405b261ecSmrg#ifdef NULL_ROOT_CURSOR 44505b261ecSmrg cm.width = 0; 44605b261ecSmrg cm.height = 0; 44705b261ecSmrg cm.xhot = 0; 44805b261ecSmrg cm.yhot = 0; 44905b261ecSmrg 45005b261ecSmrg curs = AllocCursor(NULL, NULL, &cm, 0, 0, 0, 0, 0, 0); 45105b261ecSmrg 45205b261ecSmrg if (curs == NullCursor) 45305b261ecSmrg return NullCursor; 45405b261ecSmrg#else 45505b261ecSmrg fontID = FakeClientID(0); 45605b261ecSmrg err = OpenFont(serverClient, fontID, FontLoadAll | FontOpenSync, 45705b261ecSmrg (unsigned)strlen(defaultCursorFont), defaultCursorFont); 45805b261ecSmrg if (err != Success) 45905b261ecSmrg return NullCursor; 46005b261ecSmrg 46105b261ecSmrg cursorfont = (FontPtr)LookupIDByType(fontID, RT_FONT); 46205b261ecSmrg if (!cursorfont) 46305b261ecSmrg return NullCursor; 46405b261ecSmrg if (AllocGlyphCursor(fontID, 0, fontID, 1, 46505b261ecSmrg 0, 0, 0, ~0, ~0, ~0, &curs, serverClient) != Success) 46605b261ecSmrg return NullCursor; 46705b261ecSmrg#endif 46805b261ecSmrg 46905b261ecSmrg if (!AddResource(FakeClientID(0), RT_CURSOR, (pointer)curs)) 47005b261ecSmrg return NullCursor; 47105b261ecSmrg 47205b261ecSmrg return curs; 47305b261ecSmrg} 474