/************************************************************
Copyright 1989 by The Massachusetts Institute of Technology
Permission to use, copy, modify, and distribute this
software and its documentation for any purpose and without
fee is hereby granted, provided that the above copyright
notice appear in all copies and that both that copyright
notice and this permission notice appear in supporting
documentation, and that the name of MIT not be used in
advertising or publicity pertaining to distribution of the
software without specific prior written permission.
M.I.T. makes no representation about the suitability of
this software for any purpose. It is provided "as is"
without any express or implied warranty.
********************************************************/
/* $XConsortium: mfbzerarc.c,v 5.10 89/09/20 18:55:33 rws Exp $ */
* "Algorithm for drawing ellipses or hyperbolae with a digital plotter"
* The Computer Journal, November 1967, Volume 10, Number 3, pp. 282-289
extern void miPolyArc(), miZeroPolyArc();
#if (BITMAP_BIT_ORDER == MSBFirst)
#define LEFTMOST ((unsigned int) 0x80000000)
#define LEFTMOST ((unsigned int) 1)
#define PixelateWhite(addr,off) \
(addr)[(off)>>5] |= SCRRIGHT (LEFTMOST, ((off) & 0x1f))
#define PixelateBlack(addr,off) \
(addr)[(off)>>5] &= ~(SCRRIGHT (LEFTMOST, ((off) & 0x1f)))
#define Pixelate(base,off) \
paddr = base + ((off)>>5); \
pmask = SCRRIGHT(LEFTMOST, (off) & 0x1f); \
*paddr = (*paddr & ~pmask) | (pixel & pmask); \
#define DoPix(bit,base,off) if (mask & bit) Pixelate(base,off);
mfbZeroArcSS(pDraw
, pGC
, arc
)
register int x
, y
, a
, b
, d
, mask
;
register int k1
, k3
, dx
, dy
;
int nlwidth
, yoffset
, dyoffset
;
if (((mfbPrivGC
*)(pGC
->devPrivates
[mfbGCPrivateIndex
].ptr
))->rop
==
if (pDraw
->type
== DRAWABLE_WINDOW
)
addrl
= (int *) getPrivScreenPtr(pDraw
->pScreen
)->bits
;
nlwidth
= (int) getPrivScreenPtr(pDraw
->pScreen
)->stride
>> 2;
addrl
= (int *) getPrivPixmapPtr(pDraw
)->bits
;
nlwidth
= (int) getPrivPixmapPtr(pDraw
)->stride
>> 2;
do360
= miZeroArcSetup(arc
, &info
, TRUE
);
yorgl
= addrl
+ ((info
.yorg
+ pDraw
->y
) * nlwidth
);
yorgol
= addrl
+ ((info
.yorgo
+ pDraw
->y
) * nlwidth
);
yoffset
= y
? nlwidth
: 0;
DoPix(2, yorgl
, info
.xorgo
);
DoPix(8, yorgol
, info
.xorgo
);
if (!info
.end
.x
|| !info
.end
.y
)
if (do360
&& (arc
->width
== arc
->height
) && !(arc
->width
& 1))
int *yorghl
= yorgl
+ (info
.h
* nlwidth
);
int xorghp
= info
.xorg
+ info
.h
;
int xorghn
= info
.xorg
- info
.h
;
PixelateWhite(yorgl
+ yoffset
, info
.xorg
+ x
);
PixelateWhite(yorgl
+ yoffset
, info
.xorg
- x
);
PixelateWhite(yorgol
- yoffset
, info
.xorg
- x
);
PixelateWhite(yorgol
- yoffset
, info
.xorg
+ x
);
PixelateWhite(yorghl
- xoffset
, xorghp
- y
);
PixelateWhite(yorghl
- xoffset
, xorghn
+ y
);
PixelateWhite(yorghl
+ xoffset
, xorghn
+ y
);
PixelateWhite(yorghl
+ xoffset
, xorghp
- y
);
MIARCCIRCLESTEP(yoffset
+= nlwidth
;);
PixelateBlack(yorgl
+ yoffset
, info
.xorg
+ x
);
PixelateBlack(yorgl
+ yoffset
, info
.xorg
- x
);
PixelateBlack(yorgol
- yoffset
, info
.xorg
- x
);
PixelateBlack(yorgol
- yoffset
, info
.xorg
+ x
);
PixelateBlack(yorghl
- xoffset
, xorghp
- y
);
PixelateBlack(yorghl
- xoffset
, xorghn
+ y
);
PixelateBlack(yorghl
+ xoffset
, xorghn
+ y
);
PixelateBlack(yorghl
+ xoffset
, xorghp
- y
);
MIARCCIRCLESTEP(yoffset
+= nlwidth
;);
yoffset
= info
.h
* nlwidth
;
while (y
< info
.h
|| x
< info
.w
)
MIARCOCTANTSHIFT(dyoffset
= nlwidth
;);
Pixelate(yorgl
+ yoffset
, info
.xorg
+ x
);
Pixelate(yorgl
+ yoffset
, info
.xorgo
- x
);
Pixelate(yorgol
- yoffset
, info
.xorgo
- x
);
Pixelate(yorgol
- yoffset
, info
.xorg
+ x
);
MIARCSTEP(yoffset
+= dyoffset
;, yoffset
+= nlwidth
;);
while (y
< info
.h
|| x
< info
.w
)
MIARCOCTANTSHIFT(dyoffset
= nlwidth
;);
if ((x
== info
.start
.x
) || (y
== info
.start
.y
))
info
.start
= info
.altstart
;
DoPix(1, yorgl
+ yoffset
, info
.xorg
+ x
);
DoPix(2, yorgl
+ yoffset
, info
.xorgo
- x
);
DoPix(4, yorgol
- yoffset
, info
.xorgo
- x
);
DoPix(8, yorgol
- yoffset
, info
.xorg
+ x
);
if ((x
== info
.end
.x
) || (y
== info
.end
.y
))
MIARCSTEP(yoffset
+= dyoffset
;, yoffset
+= nlwidth
;);
if ((x
== info
.start
.x
) || (y
== info
.start
.y
))
DoPix(1, yorgl
+ yoffset
, info
.xorg
+ x
);
DoPix(4, yorgol
- yoffset
, info
.xorgo
- x
);
DoPix(2, yorgl
+ yoffset
, info
.xorgo
- x
);
DoPix(8, yorgol
- yoffset
, info
.xorg
+ x
);
mfbZeroPolyArcSS(pDraw
, pGC
, narcs
, parcs
)
cclip
= ((mfbPrivGC
*)(pGC
->devPrivates
[mfbGCPrivateIndex
].ptr
))->pCompositeClip
;
for (arc
= parcs
, i
= narcs
; --i
>= 0; arc
++)
box
.x1
= arc
->x
+ pDraw
->x
;
box
.y1
= arc
->y
+ pDraw
->y
;
box
.x2
= box
.x1
+ (int)arc
->width
+ 1;
box
.y2
= box
.y1
+ (int)arc
->height
+ 1;
if ((*pDraw
->pScreen
->RectIn
)(cclip
, &box
) == rgnIN
)
mfbZeroArcSS(pDraw
, pGC
, arc
);
miZeroPolyArc(pDraw
, pGC
, 1, arc
);
miPolyArc(pDraw
, pGC
, 1, arc
);