1*4882a593Smuzhiyun #ifndef __ATI_RADEON_FB_H
2*4882a593Smuzhiyun #define __ATI_RADEON_FB_H
3*4882a593Smuzhiyun
4*4882a593Smuzhiyun /***************************************************************
5*4882a593Smuzhiyun * Most of the definitions here are adapted right from XFree86 *
6*4882a593Smuzhiyun ***************************************************************/
7*4882a593Smuzhiyun
8*4882a593Smuzhiyun /*
9*4882a593Smuzhiyun * Chip families. Must fit in the low 16 bits of a long word
10*4882a593Smuzhiyun */
11*4882a593Smuzhiyun enum radeon_family {
12*4882a593Smuzhiyun CHIP_FAMILY_UNKNOW,
13*4882a593Smuzhiyun CHIP_FAMILY_LEGACY,
14*4882a593Smuzhiyun CHIP_FAMILY_RADEON,
15*4882a593Smuzhiyun CHIP_FAMILY_RV100,
16*4882a593Smuzhiyun CHIP_FAMILY_RS100, /* U1 (IGP320M) or A3 (IGP320)*/
17*4882a593Smuzhiyun CHIP_FAMILY_RV200,
18*4882a593Smuzhiyun CHIP_FAMILY_RS200, /* U2 (IGP330M/340M/350M) or A4 (IGP330/340/345/350),
19*4882a593Smuzhiyun RS250 (IGP 7000) */
20*4882a593Smuzhiyun CHIP_FAMILY_R200,
21*4882a593Smuzhiyun CHIP_FAMILY_RV250,
22*4882a593Smuzhiyun CHIP_FAMILY_RS300, /* Radeon 9000 IGP */
23*4882a593Smuzhiyun CHIP_FAMILY_RV280,
24*4882a593Smuzhiyun CHIP_FAMILY_R300,
25*4882a593Smuzhiyun CHIP_FAMILY_R350,
26*4882a593Smuzhiyun CHIP_FAMILY_RV350,
27*4882a593Smuzhiyun CHIP_FAMILY_RV380, /* RV370/RV380/M22/M24 */
28*4882a593Smuzhiyun CHIP_FAMILY_R420, /* R420/R423/M18 */
29*4882a593Smuzhiyun CHIP_FAMILY_LAST,
30*4882a593Smuzhiyun };
31*4882a593Smuzhiyun
32*4882a593Smuzhiyun #define IS_RV100_VARIANT(rinfo) (((rinfo)->family == CHIP_FAMILY_RV100) || \
33*4882a593Smuzhiyun ((rinfo)->family == CHIP_FAMILY_RV200) || \
34*4882a593Smuzhiyun ((rinfo)->family == CHIP_FAMILY_RS100) || \
35*4882a593Smuzhiyun ((rinfo)->family == CHIP_FAMILY_RS200) || \
36*4882a593Smuzhiyun ((rinfo)->family == CHIP_FAMILY_RV250) || \
37*4882a593Smuzhiyun ((rinfo)->family == CHIP_FAMILY_RV280) || \
38*4882a593Smuzhiyun ((rinfo)->family == CHIP_FAMILY_RS300))
39*4882a593Smuzhiyun
40*4882a593Smuzhiyun #define IS_R300_VARIANT(rinfo) (((rinfo)->family == CHIP_FAMILY_R300) || \
41*4882a593Smuzhiyun ((rinfo)->family == CHIP_FAMILY_RV350) || \
42*4882a593Smuzhiyun ((rinfo)->family == CHIP_FAMILY_R350) || \
43*4882a593Smuzhiyun ((rinfo)->family == CHIP_FAMILY_RV380) || \
44*4882a593Smuzhiyun ((rinfo)->family == CHIP_FAMILY_R420))
45*4882a593Smuzhiyun
46*4882a593Smuzhiyun struct radeonfb_info {
47*4882a593Smuzhiyun char name[20];
48*4882a593Smuzhiyun
49*4882a593Smuzhiyun struct pci_device_id pdev;
50*4882a593Smuzhiyun u16 family;
51*4882a593Smuzhiyun
52*4882a593Smuzhiyun u32 fb_base_bus;
53*4882a593Smuzhiyun u32 mmio_base_bus;
54*4882a593Smuzhiyun
55*4882a593Smuzhiyun void *mmio_base;
56*4882a593Smuzhiyun void *fb_base;
57*4882a593Smuzhiyun
58*4882a593Smuzhiyun u32 video_ram;
59*4882a593Smuzhiyun u32 mapped_vram;
60*4882a593Smuzhiyun int vram_width;
61*4882a593Smuzhiyun int vram_ddr;
62*4882a593Smuzhiyun
63*4882a593Smuzhiyun u32 fb_local_base;
64*4882a593Smuzhiyun };
65*4882a593Smuzhiyun
66*4882a593Smuzhiyun #define INREG8(addr) readb((rinfo->mmio_base)+addr)
67*4882a593Smuzhiyun #define OUTREG8(addr,val) writeb(val, (rinfo->mmio_base)+addr)
68*4882a593Smuzhiyun #define INREG16(addr) readw((rinfo->mmio_base)+addr)
69*4882a593Smuzhiyun #define OUTREG16(addr,val) writew(val, (rinfo->mmio_base)+addr)
70*4882a593Smuzhiyun #define INREG(addr) readl((rinfo->mmio_base)+addr)
71*4882a593Smuzhiyun #define OUTREG(addr,val) writel(val, (rinfo->mmio_base)+addr)
72*4882a593Smuzhiyun
_OUTREGP(struct radeonfb_info * rinfo,u32 addr,u32 val,u32 mask)73*4882a593Smuzhiyun static inline void _OUTREGP(struct radeonfb_info *rinfo, u32 addr,
74*4882a593Smuzhiyun u32 val, u32 mask)
75*4882a593Smuzhiyun {
76*4882a593Smuzhiyun unsigned int tmp;
77*4882a593Smuzhiyun
78*4882a593Smuzhiyun tmp = INREG(addr);
79*4882a593Smuzhiyun tmp &= (mask);
80*4882a593Smuzhiyun tmp |= (val);
81*4882a593Smuzhiyun OUTREG(addr, tmp);
82*4882a593Smuzhiyun }
83*4882a593Smuzhiyun
84*4882a593Smuzhiyun #define OUTREGP(addr,val,mask) _OUTREGP(rinfo, addr, val,mask)
85*4882a593Smuzhiyun
86*4882a593Smuzhiyun /*
87*4882a593Smuzhiyun * 2D Engine helper routines
88*4882a593Smuzhiyun */
radeon_engine_flush(struct radeonfb_info * rinfo)89*4882a593Smuzhiyun static inline void radeon_engine_flush (struct radeonfb_info *rinfo)
90*4882a593Smuzhiyun {
91*4882a593Smuzhiyun int i;
92*4882a593Smuzhiyun
93*4882a593Smuzhiyun /* initiate flush */
94*4882a593Smuzhiyun OUTREGP(RB2D_DSTCACHE_CTLSTAT, RB2D_DC_FLUSH_ALL,
95*4882a593Smuzhiyun ~RB2D_DC_FLUSH_ALL);
96*4882a593Smuzhiyun
97*4882a593Smuzhiyun for (i=0; i < 2000000; i++) {
98*4882a593Smuzhiyun if (!(INREG(RB2D_DSTCACHE_CTLSTAT) & RB2D_DC_BUSY))
99*4882a593Smuzhiyun return;
100*4882a593Smuzhiyun udelay(1);
101*4882a593Smuzhiyun }
102*4882a593Smuzhiyun printf("radeonfb: Flush Timeout !\n");
103*4882a593Smuzhiyun }
104*4882a593Smuzhiyun
_radeon_fifo_wait(struct radeonfb_info * rinfo,int entries)105*4882a593Smuzhiyun static inline void _radeon_fifo_wait(struct radeonfb_info *rinfo, int entries)
106*4882a593Smuzhiyun {
107*4882a593Smuzhiyun int i;
108*4882a593Smuzhiyun
109*4882a593Smuzhiyun for (i=0; i<2000000; i++) {
110*4882a593Smuzhiyun if ((INREG(RBBM_STATUS) & 0x7f) >= entries)
111*4882a593Smuzhiyun return;
112*4882a593Smuzhiyun udelay(1);
113*4882a593Smuzhiyun }
114*4882a593Smuzhiyun printf("radeonfb: FIFO Timeout !\n");
115*4882a593Smuzhiyun }
116*4882a593Smuzhiyun
_radeon_engine_idle(struct radeonfb_info * rinfo)117*4882a593Smuzhiyun static inline void _radeon_engine_idle(struct radeonfb_info *rinfo)
118*4882a593Smuzhiyun {
119*4882a593Smuzhiyun int i;
120*4882a593Smuzhiyun
121*4882a593Smuzhiyun /* ensure FIFO is empty before waiting for idle */
122*4882a593Smuzhiyun _radeon_fifo_wait (rinfo, 64);
123*4882a593Smuzhiyun
124*4882a593Smuzhiyun for (i=0; i<2000000; i++) {
125*4882a593Smuzhiyun if (((INREG(RBBM_STATUS) & GUI_ACTIVE)) == 0) {
126*4882a593Smuzhiyun radeon_engine_flush (rinfo);
127*4882a593Smuzhiyun return;
128*4882a593Smuzhiyun }
129*4882a593Smuzhiyun udelay(1);
130*4882a593Smuzhiyun }
131*4882a593Smuzhiyun printf("radeonfb: Idle Timeout !\n");
132*4882a593Smuzhiyun }
133*4882a593Smuzhiyun
134*4882a593Smuzhiyun #define radeon_engine_idle() _radeon_engine_idle(rinfo)
135*4882a593Smuzhiyun #define radeon_fifo_wait(entries) _radeon_fifo_wait(rinfo,entries)
136*4882a593Smuzhiyun #define radeon_msleep(ms) _radeon_msleep(rinfo,ms)
137*4882a593Smuzhiyun
138*4882a593Smuzhiyun /*
139*4882a593Smuzhiyun * This structure contains the various registers manipulated by this
140*4882a593Smuzhiyun * driver for setting or restoring a mode. It's mostly copied from
141*4882a593Smuzhiyun * XFree's RADEONSaveRec structure. A few chip settings might still be
142*4882a593Smuzhiyun * tweaked without beeing reflected or saved in these registers though
143*4882a593Smuzhiyun */
144*4882a593Smuzhiyun struct radeon_regs {
145*4882a593Smuzhiyun /* Common registers */
146*4882a593Smuzhiyun u32 ovr_clr;
147*4882a593Smuzhiyun u32 ovr_wid_left_right;
148*4882a593Smuzhiyun u32 ovr_wid_top_bottom;
149*4882a593Smuzhiyun u32 ov0_scale_cntl;
150*4882a593Smuzhiyun u32 mpp_tb_config;
151*4882a593Smuzhiyun u32 mpp_gp_config;
152*4882a593Smuzhiyun u32 subpic_cntl;
153*4882a593Smuzhiyun u32 viph_control;
154*4882a593Smuzhiyun u32 i2c_cntl_1;
155*4882a593Smuzhiyun u32 gen_int_cntl;
156*4882a593Smuzhiyun u32 cap0_trig_cntl;
157*4882a593Smuzhiyun u32 cap1_trig_cntl;
158*4882a593Smuzhiyun u32 bus_cntl;
159*4882a593Smuzhiyun u32 surface_cntl;
160*4882a593Smuzhiyun u32 bios_5_scratch;
161*4882a593Smuzhiyun
162*4882a593Smuzhiyun /* Other registers to save for VT switches or driver load/unload */
163*4882a593Smuzhiyun u32 dp_datatype;
164*4882a593Smuzhiyun u32 rbbm_soft_reset;
165*4882a593Smuzhiyun u32 clock_cntl_index;
166*4882a593Smuzhiyun u32 amcgpio_en_reg;
167*4882a593Smuzhiyun u32 amcgpio_mask;
168*4882a593Smuzhiyun
169*4882a593Smuzhiyun /* Surface/tiling registers */
170*4882a593Smuzhiyun u32 surf_lower_bound[8];
171*4882a593Smuzhiyun u32 surf_upper_bound[8];
172*4882a593Smuzhiyun u32 surf_info[8];
173*4882a593Smuzhiyun
174*4882a593Smuzhiyun /* CRTC registers */
175*4882a593Smuzhiyun u32 crtc_gen_cntl;
176*4882a593Smuzhiyun u32 crtc_ext_cntl;
177*4882a593Smuzhiyun u32 dac_cntl;
178*4882a593Smuzhiyun u32 crtc_h_total_disp;
179*4882a593Smuzhiyun u32 crtc_h_sync_strt_wid;
180*4882a593Smuzhiyun u32 crtc_v_total_disp;
181*4882a593Smuzhiyun u32 crtc_v_sync_strt_wid;
182*4882a593Smuzhiyun u32 crtc_offset;
183*4882a593Smuzhiyun u32 crtc_offset_cntl;
184*4882a593Smuzhiyun u32 crtc_pitch;
185*4882a593Smuzhiyun u32 disp_merge_cntl;
186*4882a593Smuzhiyun u32 grph_buffer_cntl;
187*4882a593Smuzhiyun u32 crtc_more_cntl;
188*4882a593Smuzhiyun
189*4882a593Smuzhiyun /* CRTC2 registers */
190*4882a593Smuzhiyun u32 crtc2_gen_cntl;
191*4882a593Smuzhiyun u32 dac2_cntl;
192*4882a593Smuzhiyun u32 disp_output_cntl;
193*4882a593Smuzhiyun u32 disp_hw_debug;
194*4882a593Smuzhiyun u32 disp2_merge_cntl;
195*4882a593Smuzhiyun u32 grph2_buffer_cntl;
196*4882a593Smuzhiyun u32 crtc2_h_total_disp;
197*4882a593Smuzhiyun u32 crtc2_h_sync_strt_wid;
198*4882a593Smuzhiyun u32 crtc2_v_total_disp;
199*4882a593Smuzhiyun u32 crtc2_v_sync_strt_wid;
200*4882a593Smuzhiyun u32 crtc2_offset;
201*4882a593Smuzhiyun u32 crtc2_offset_cntl;
202*4882a593Smuzhiyun u32 crtc2_pitch;
203*4882a593Smuzhiyun
204*4882a593Smuzhiyun /* Flat panel regs */
205*4882a593Smuzhiyun u32 fp_crtc_h_total_disp;
206*4882a593Smuzhiyun u32 fp_crtc_v_total_disp;
207*4882a593Smuzhiyun u32 fp_gen_cntl;
208*4882a593Smuzhiyun u32 fp2_gen_cntl;
209*4882a593Smuzhiyun u32 fp_h_sync_strt_wid;
210*4882a593Smuzhiyun u32 fp2_h_sync_strt_wid;
211*4882a593Smuzhiyun u32 fp_horz_stretch;
212*4882a593Smuzhiyun u32 fp_panel_cntl;
213*4882a593Smuzhiyun u32 fp_v_sync_strt_wid;
214*4882a593Smuzhiyun u32 fp2_v_sync_strt_wid;
215*4882a593Smuzhiyun u32 fp_vert_stretch;
216*4882a593Smuzhiyun u32 lvds_gen_cntl;
217*4882a593Smuzhiyun u32 lvds_pll_cntl;
218*4882a593Smuzhiyun u32 tmds_crc;
219*4882a593Smuzhiyun u32 tmds_transmitter_cntl;
220*4882a593Smuzhiyun
221*4882a593Smuzhiyun /* Computed values for PLL */
222*4882a593Smuzhiyun u32 dot_clock_freq;
223*4882a593Smuzhiyun int feedback_div;
224*4882a593Smuzhiyun int post_div;
225*4882a593Smuzhiyun
226*4882a593Smuzhiyun /* PLL registers */
227*4882a593Smuzhiyun u32 ppll_div_3;
228*4882a593Smuzhiyun u32 ppll_ref_div;
229*4882a593Smuzhiyun u32 vclk_ecp_cntl;
230*4882a593Smuzhiyun u32 clk_cntl_index;
231*4882a593Smuzhiyun
232*4882a593Smuzhiyun /* Computed values for PLL2 */
233*4882a593Smuzhiyun u32 dot_clock_freq_2;
234*4882a593Smuzhiyun int feedback_div_2;
235*4882a593Smuzhiyun int post_div_2;
236*4882a593Smuzhiyun
237*4882a593Smuzhiyun /* PLL2 registers */
238*4882a593Smuzhiyun u32 p2pll_ref_div;
239*4882a593Smuzhiyun u32 p2pll_div_0;
240*4882a593Smuzhiyun u32 htotal_cntl2;
241*4882a593Smuzhiyun
242*4882a593Smuzhiyun /* Palette */
243*4882a593Smuzhiyun int palette_valid;
244*4882a593Smuzhiyun };
245*4882a593Smuzhiyun
__INPLL(struct radeonfb_info * rinfo,u32 addr)246*4882a593Smuzhiyun static inline u32 __INPLL(struct radeonfb_info *rinfo, u32 addr)
247*4882a593Smuzhiyun {
248*4882a593Smuzhiyun u32 data;
249*4882a593Smuzhiyun
250*4882a593Smuzhiyun OUTREG8(CLOCK_CNTL_INDEX, addr & 0x0000003f);
251*4882a593Smuzhiyun /* radeon_pll_errata_after_index(rinfo); */
252*4882a593Smuzhiyun data = INREG(CLOCK_CNTL_DATA);
253*4882a593Smuzhiyun /* radeon_pll_errata_after_data(rinfo); */
254*4882a593Smuzhiyun return data;
255*4882a593Smuzhiyun }
256*4882a593Smuzhiyun
__OUTPLL(struct radeonfb_info * rinfo,unsigned int index,u32 val)257*4882a593Smuzhiyun static inline void __OUTPLL(struct radeonfb_info *rinfo, unsigned int index,
258*4882a593Smuzhiyun u32 val)
259*4882a593Smuzhiyun {
260*4882a593Smuzhiyun
261*4882a593Smuzhiyun OUTREG8(CLOCK_CNTL_INDEX, (index & 0x0000003f) | 0x00000080);
262*4882a593Smuzhiyun /* radeon_pll_errata_after_index(rinfo); */
263*4882a593Smuzhiyun OUTREG(CLOCK_CNTL_DATA, val);
264*4882a593Smuzhiyun /* radeon_pll_errata_after_data(rinfo); */
265*4882a593Smuzhiyun }
266*4882a593Smuzhiyun
__OUTPLLP(struct radeonfb_info * rinfo,unsigned int index,u32 val,u32 mask)267*4882a593Smuzhiyun static inline void __OUTPLLP(struct radeonfb_info *rinfo, unsigned int index,
268*4882a593Smuzhiyun u32 val, u32 mask)
269*4882a593Smuzhiyun {
270*4882a593Smuzhiyun unsigned int tmp;
271*4882a593Smuzhiyun
272*4882a593Smuzhiyun tmp = __INPLL(rinfo, index);
273*4882a593Smuzhiyun tmp &= (mask);
274*4882a593Smuzhiyun tmp |= (val);
275*4882a593Smuzhiyun __OUTPLL(rinfo, index, tmp);
276*4882a593Smuzhiyun }
277*4882a593Smuzhiyun
278*4882a593Smuzhiyun #define INPLL(addr) __INPLL(rinfo, addr)
279*4882a593Smuzhiyun #define OUTPLL(index, val) __OUTPLL(rinfo, index, val)
280*4882a593Smuzhiyun #define OUTPLLP(index, val, mask) __OUTPLLP(rinfo, index, val, mask)
281*4882a593Smuzhiyun
282*4882a593Smuzhiyun #endif
283