1*4882a593Smuzhiyun /************************************************************************
2*4882a593Smuzhiyun
3*4882a593Smuzhiyun Copyright 1987, 1998 The Open Group
4*4882a593Smuzhiyun
5*4882a593Smuzhiyun Permission to use, copy, modify, distribute, and sell this software and its
6*4882a593Smuzhiyun documentation for any purpose is hereby granted without fee, provided that
7*4882a593Smuzhiyun the above copyright notice appear in all copies and that both that
8*4882a593Smuzhiyun copyright notice and this permission notice appear in supporting
9*4882a593Smuzhiyun documentation.
10*4882a593Smuzhiyun
11*4882a593Smuzhiyun The above copyright notice and this permission notice shall be included in
12*4882a593Smuzhiyun all copies or substantial portions of the Software.
13*4882a593Smuzhiyun
14*4882a593Smuzhiyun THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15*4882a593Smuzhiyun IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16*4882a593Smuzhiyun FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
17*4882a593Smuzhiyun OPEN GROUP BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
18*4882a593Smuzhiyun AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
19*4882a593Smuzhiyun CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
20*4882a593Smuzhiyun
21*4882a593Smuzhiyun Except as contained in this notice, the name of The Open Group shall not be
22*4882a593Smuzhiyun used in advertising or otherwise to promote the sale, use or other dealings
23*4882a593Smuzhiyun in this Software without prior written authorization from The Open Group.
24*4882a593Smuzhiyun
25*4882a593Smuzhiyun Copyright 1987 by Digital Equipment Corporation, Maynard, Massachusetts.
26*4882a593Smuzhiyun
27*4882a593Smuzhiyun All Rights Reserved
28*4882a593Smuzhiyun
29*4882a593Smuzhiyun Permission to use, copy, modify, and distribute this software and its
30*4882a593Smuzhiyun documentation for any purpose and without fee is hereby granted,
31*4882a593Smuzhiyun provided that the above copyright notice appear in all copies and that
32*4882a593Smuzhiyun both that copyright notice and this permission notice appear in
33*4882a593Smuzhiyun supporting documentation, and that the name of Digital not be
34*4882a593Smuzhiyun used in advertising or publicity pertaining to distribution of the
35*4882a593Smuzhiyun software without specific, written prior permission.
36*4882a593Smuzhiyun
37*4882a593Smuzhiyun DIGITAL DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING
38*4882a593Smuzhiyun ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL
39*4882a593Smuzhiyun DIGITAL BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR
40*4882a593Smuzhiyun ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS,
41*4882a593Smuzhiyun WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION,
42*4882a593Smuzhiyun ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
43*4882a593Smuzhiyun SOFTWARE.
44*4882a593Smuzhiyun
45*4882a593Smuzhiyun ************************************************************************/
46*4882a593Smuzhiyun
47*4882a593Smuzhiyun #ifdef HAVE_DIX_CONFIG_H
48*4882a593Smuzhiyun #include <dix-config.h>
49*4882a593Smuzhiyun #endif
50*4882a593Smuzhiyun
51*4882a593Smuzhiyun #include "misc.h"
52*4882a593Smuzhiyun #include <X11/fonts/fontstruct.h>
53*4882a593Smuzhiyun #include "dixfontstr.h"
54*4882a593Smuzhiyun #include "scrnintstr.h"
55*4882a593Smuzhiyun #include "gcstruct.h"
56*4882a593Smuzhiyun #include "resource.h"
57*4882a593Smuzhiyun #include "dix.h"
58*4882a593Smuzhiyun #include "cursorstr.h"
59*4882a593Smuzhiyun #include "opaque.h"
60*4882a593Smuzhiyun #include "servermd.h"
61*4882a593Smuzhiyun
62*4882a593Smuzhiyun /*
63*4882a593Smuzhiyun get the bits out of the font in a portable way. to avoid
64*4882a593Smuzhiyun dealing with padding and such-like, we draw the glyph into
65*4882a593Smuzhiyun a bitmap, then read the bits out with GetImage, which
66*4882a593Smuzhiyun uses server-natural format.
67*4882a593Smuzhiyun since all screens return the same bitmap format, we'll just use
68*4882a593Smuzhiyun the first one we find.
69*4882a593Smuzhiyun the character origin lines up with the hotspot in the
70*4882a593Smuzhiyun cursor metrics.
71*4882a593Smuzhiyun */
72*4882a593Smuzhiyun
73*4882a593Smuzhiyun int
ServerBitsFromGlyph(FontPtr pfont,unsigned ch,CursorMetricPtr cm,unsigned char ** ppbits)74*4882a593Smuzhiyun ServerBitsFromGlyph(FontPtr pfont, unsigned ch, CursorMetricPtr cm,
75*4882a593Smuzhiyun unsigned char **ppbits)
76*4882a593Smuzhiyun {
77*4882a593Smuzhiyun ScreenPtr pScreen;
78*4882a593Smuzhiyun GCPtr pGC;
79*4882a593Smuzhiyun xRectangle rect;
80*4882a593Smuzhiyun PixmapPtr ppix;
81*4882a593Smuzhiyun char *pbits;
82*4882a593Smuzhiyun ChangeGCVal gcval[3];
83*4882a593Smuzhiyun unsigned char char2b[2];
84*4882a593Smuzhiyun
85*4882a593Smuzhiyun /* turn glyph index into a protocol-format char2b */
86*4882a593Smuzhiyun char2b[0] = (unsigned char) (ch >> 8);
87*4882a593Smuzhiyun char2b[1] = (unsigned char) (ch & 0xff);
88*4882a593Smuzhiyun
89*4882a593Smuzhiyun pScreen = screenInfo.screens[0];
90*4882a593Smuzhiyun pbits = calloc(BitmapBytePad(cm->width), cm->height);
91*4882a593Smuzhiyun if (!pbits)
92*4882a593Smuzhiyun return BadAlloc;
93*4882a593Smuzhiyun
94*4882a593Smuzhiyun ppix = (PixmapPtr) (*pScreen->CreatePixmap) (pScreen, cm->width,
95*4882a593Smuzhiyun cm->height, 1,
96*4882a593Smuzhiyun CREATE_PIXMAP_USAGE_SCRATCH);
97*4882a593Smuzhiyun pGC = GetScratchGC(1, pScreen);
98*4882a593Smuzhiyun if (!ppix || !pGC) {
99*4882a593Smuzhiyun if (ppix)
100*4882a593Smuzhiyun (*pScreen->DestroyPixmap) (ppix);
101*4882a593Smuzhiyun if (pGC)
102*4882a593Smuzhiyun FreeScratchGC(pGC);
103*4882a593Smuzhiyun free(pbits);
104*4882a593Smuzhiyun return BadAlloc;
105*4882a593Smuzhiyun }
106*4882a593Smuzhiyun
107*4882a593Smuzhiyun rect.x = 0;
108*4882a593Smuzhiyun rect.y = 0;
109*4882a593Smuzhiyun rect.width = cm->width;
110*4882a593Smuzhiyun rect.height = cm->height;
111*4882a593Smuzhiyun
112*4882a593Smuzhiyun /* fill the pixmap with 0 */
113*4882a593Smuzhiyun gcval[0].val = GXcopy;
114*4882a593Smuzhiyun gcval[1].val = 0;
115*4882a593Smuzhiyun gcval[2].ptr = (void *) pfont;
116*4882a593Smuzhiyun ChangeGC(NullClient, pGC, GCFunction | GCForeground | GCFont, gcval);
117*4882a593Smuzhiyun ValidateGC((DrawablePtr) ppix, pGC);
118*4882a593Smuzhiyun (*pGC->ops->PolyFillRect) ((DrawablePtr) ppix, pGC, 1, &rect);
119*4882a593Smuzhiyun
120*4882a593Smuzhiyun /* draw the glyph */
121*4882a593Smuzhiyun gcval[0].val = 1;
122*4882a593Smuzhiyun ChangeGC(NullClient, pGC, GCForeground, gcval);
123*4882a593Smuzhiyun ValidateGC((DrawablePtr) ppix, pGC);
124*4882a593Smuzhiyun (*pGC->ops->PolyText16) ((DrawablePtr) ppix, pGC, cm->xhot, cm->yhot,
125*4882a593Smuzhiyun 1, (unsigned short *) char2b);
126*4882a593Smuzhiyun (*pScreen->GetImage) ((DrawablePtr) ppix, 0, 0, cm->width, cm->height,
127*4882a593Smuzhiyun XYPixmap, 1, pbits);
128*4882a593Smuzhiyun *ppbits = (unsigned char *) pbits;
129*4882a593Smuzhiyun FreeScratchGC(pGC);
130*4882a593Smuzhiyun (*pScreen->DestroyPixmap) (ppix);
131*4882a593Smuzhiyun return Success;
132*4882a593Smuzhiyun }
133*4882a593Smuzhiyun
134*4882a593Smuzhiyun Bool
CursorMetricsFromGlyph(FontPtr pfont,unsigned ch,CursorMetricPtr cm)135*4882a593Smuzhiyun CursorMetricsFromGlyph(FontPtr pfont, unsigned ch, CursorMetricPtr cm)
136*4882a593Smuzhiyun {
137*4882a593Smuzhiyun CharInfoPtr pci;
138*4882a593Smuzhiyun unsigned long nglyphs;
139*4882a593Smuzhiyun CARD8 chs[2];
140*4882a593Smuzhiyun FontEncoding encoding;
141*4882a593Smuzhiyun
142*4882a593Smuzhiyun chs[0] = ch >> 8;
143*4882a593Smuzhiyun chs[1] = ch;
144*4882a593Smuzhiyun encoding = (FONTLASTROW(pfont) == 0) ? Linear16Bit : TwoD16Bit;
145*4882a593Smuzhiyun if (encoding == Linear16Bit) {
146*4882a593Smuzhiyun if (ch < pfont->info.firstCol || pfont->info.lastCol < ch)
147*4882a593Smuzhiyun return FALSE;
148*4882a593Smuzhiyun }
149*4882a593Smuzhiyun else {
150*4882a593Smuzhiyun if (chs[0] < pfont->info.firstRow || pfont->info.lastRow < chs[0])
151*4882a593Smuzhiyun return FALSE;
152*4882a593Smuzhiyun if (chs[1] < pfont->info.firstCol || pfont->info.lastCol < chs[1])
153*4882a593Smuzhiyun return FALSE;
154*4882a593Smuzhiyun }
155*4882a593Smuzhiyun (*pfont->get_glyphs) (pfont, 1, chs, encoding, &nglyphs, &pci);
156*4882a593Smuzhiyun if (nglyphs == 0)
157*4882a593Smuzhiyun return FALSE;
158*4882a593Smuzhiyun cm->width = pci->metrics.rightSideBearing - pci->metrics.leftSideBearing;
159*4882a593Smuzhiyun cm->height = pci->metrics.descent + pci->metrics.ascent;
160*4882a593Smuzhiyun if (pci->metrics.leftSideBearing > 0) {
161*4882a593Smuzhiyun cm->width += pci->metrics.leftSideBearing;
162*4882a593Smuzhiyun cm->xhot = 0;
163*4882a593Smuzhiyun }
164*4882a593Smuzhiyun else {
165*4882a593Smuzhiyun cm->xhot = -pci->metrics.leftSideBearing;
166*4882a593Smuzhiyun if (pci->metrics.rightSideBearing < 0)
167*4882a593Smuzhiyun cm->width -= pci->metrics.rightSideBearing;
168*4882a593Smuzhiyun }
169*4882a593Smuzhiyun if (pci->metrics.ascent < 0) {
170*4882a593Smuzhiyun cm->height -= pci->metrics.ascent;
171*4882a593Smuzhiyun cm->yhot = 0;
172*4882a593Smuzhiyun }
173*4882a593Smuzhiyun else {
174*4882a593Smuzhiyun cm->yhot = pci->metrics.ascent;
175*4882a593Smuzhiyun if (pci->metrics.descent < 0)
176*4882a593Smuzhiyun cm->height -= pci->metrics.descent;
177*4882a593Smuzhiyun }
178*4882a593Smuzhiyun return TRUE;
179*4882a593Smuzhiyun }
180