1428d7b3dSmrg/**************************************************************************
2428d7b3dSmrg
3428d7b3dSmrgCopyright 1998-1999 Precision Insight, Inc., Cedar Park, Texas.
4428d7b3dSmrgCopyright © 2002 David Dawes
5428d7b3dSmrg
6428d7b3dSmrgAll Rights Reserved.
7428d7b3dSmrg
8428d7b3dSmrgPermission is hereby granted, free of charge, to any person obtaining a
9428d7b3dSmrgcopy of this software and associated documentation files (the
10428d7b3dSmrg"Software"), to deal in the Software without restriction, including
11428d7b3dSmrgwithout limitation the rights to use, copy, modify, merge, publish,
12428d7b3dSmrgdistribute, sub license, and/or sell copies of the Software, and to
13428d7b3dSmrgpermit persons to whom the Software is furnished to do so, subject to
14428d7b3dSmrgthe following conditions:
15428d7b3dSmrg
16428d7b3dSmrgThe above copyright notice and this permission notice (including the
17428d7b3dSmrgnext paragraph) shall be included in all copies or substantial portions
18428d7b3dSmrgof the Software.
19428d7b3dSmrg
20428d7b3dSmrgTHE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
21428d7b3dSmrgOR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
22428d7b3dSmrgMERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT.
23428d7b3dSmrgIN NO EVENT SHALL PRECISION INSIGHT AND/OR ITS SUPPLIERS BE LIABLE FOR
24428d7b3dSmrgANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
25428d7b3dSmrgTORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
26428d7b3dSmrgSOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
27428d7b3dSmrg
28428d7b3dSmrg**************************************************************************/
29428d7b3dSmrg
30428d7b3dSmrg#ifndef _INTEL_RING_H
31428d7b3dSmrg#define _INTEL_RING_H
32428d7b3dSmrg
33428d7b3dSmrg#define OUT_RING(n) do {						\
34428d7b3dSmrg    if (I810_DEBUG & DEBUG_VERBOSE_RING)				\
35428d7b3dSmrg	ErrorF("OUT_RING %lx: %x, (mask %x)\n",				\
36428d7b3dSmrg	       (unsigned long)(outring), (unsigned int)(n), ringmask);	\
37428d7b3dSmrg    *(volatile unsigned int *)(virt + outring) = n;			\
38428d7b3dSmrg    outring += 4; ringused += 4;					\
39428d7b3dSmrg    outring &= ringmask;						\
40428d7b3dSmrg} while (0)
41428d7b3dSmrg
42428d7b3dSmrg#define ADVANCE_LP_RING() do {						\
43428d7b3dSmrg    if (ringused > needed)						\
44428d7b3dSmrg	FatalError("%s: ADVANCE_LP_RING: exceeded allocation %d/%d\n ",	\
45428d7b3dSmrg		   __FUNCTION__, ringused, needed);   			\
46428d7b3dSmrg    else if (ringused < needed)						\
47428d7b3dSmrg	FatalError("%s: ADVANCE_LP_RING: under-used allocation %d/%d\n ", \
48428d7b3dSmrg		   __FUNCTION__, ringused, needed);   			\
49428d7b3dSmrg    pI810->LpRing->tail = outring;					\
50428d7b3dSmrg    pI810->LpRing->space -= ringused;					\
51428d7b3dSmrg    if (outring & 0x07)							\
52428d7b3dSmrg	FatalError("%s: ADVANCE_LP_RING: "				\
53428d7b3dSmrg		   "outring (0x%x) isn't on a QWord boundary\n",	\
54428d7b3dSmrg		   __FUNCTION__, outring);				\
55428d7b3dSmrg    OUTREG(LP_RING + RING_TAIL, outring);				\
56428d7b3dSmrg} while (0)
57428d7b3dSmrg
58428d7b3dSmrg/*
59428d7b3dSmrg * XXX Note: the head/tail masks are different for 810 and i830.
60428d7b3dSmrg * If the i810 always sets the higher bits to 0, then this shouldn't be
61428d7b3dSmrg * a problem.  Check this!
62428d7b3dSmrg */
63428d7b3dSmrg#define DO_RING_IDLE() do {						\
64428d7b3dSmrg    int _head;								\
65428d7b3dSmrg    int _tail;								\
66428d7b3dSmrg    do {								\
67428d7b3dSmrg	_head = INREG(LP_RING + RING_HEAD) & I830_HEAD_MASK;		\
68428d7b3dSmrg	_tail = INREG(LP_RING + RING_TAIL) & I830_TAIL_MASK;		\
69428d7b3dSmrg	DELAY(10);							\
70428d7b3dSmrg    } while (_head != _tail);						\
71428d7b3dSmrg} while( 0)
72428d7b3dSmrg
73428d7b3dSmrg#define BEGIN_LP_RING(n)						\
74428d7b3dSmrg    unsigned int outring, ringmask, ringused = 0;			\
75428d7b3dSmrg    volatile unsigned char *virt;					\
76428d7b3dSmrg    unsigned needed;								\
77428d7b3dSmrg    if ((n) & 1)							\
78428d7b3dSmrg	ErrorF("BEGIN_LP_RING called with odd argument: %d\n", n);	\
79428d7b3dSmrg    if ((n) > 2 && (I810_DEBUG&DEBUG_ALWAYS_SYNC))			\
80428d7b3dSmrg	DO_RING_IDLE();							\
81428d7b3dSmrg    needed = (n) * 4;							\
82428d7b3dSmrg    if (pI810->LpRing->space < needed)					\
83428d7b3dSmrg	WaitRingFunc(pScrn, needed, 0);					\
84428d7b3dSmrg    outring = pI810->LpRing->tail;					\
85428d7b3dSmrg    ringmask = pI810->LpRing->tail_mask;				\
86428d7b3dSmrg    virt = pI810->LpRing->virtual_start;				\
87428d7b3dSmrg    if (I810_DEBUG & DEBUG_VERBOSE_RING)				\
88428d7b3dSmrg	ErrorF( "BEGIN_LP_RING %d in %s\n", n, FUNCTION_NAME);
89428d7b3dSmrg
90428d7b3dSmrg#endif /* _INTEL_RING_H */
91