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