1209ff23fSmrg/*
2209ff23fSmrg * Copyright 2000 ATI Technologies Inc., Markham, Ontario, and
3209ff23fSmrg *                VA Linux Systems Inc., Fremont, California.
4209ff23fSmrg *
5209ff23fSmrg * All Rights Reserved.
6209ff23fSmrg *
7209ff23fSmrg * Permission is hereby granted, free of charge, to any person obtaining
8209ff23fSmrg * a copy of this software and associated documentation files (the
9209ff23fSmrg * "Software"), to deal in the Software without restriction, including
10209ff23fSmrg * without limitation on the rights to use, copy, modify, merge,
11209ff23fSmrg * publish, distribute, sublicense, and/or sell copies of the Software,
12209ff23fSmrg * and to permit persons to whom the Software is furnished to do so,
13209ff23fSmrg * subject to the following conditions:
14209ff23fSmrg *
15209ff23fSmrg * The above copyright notice and this permission notice (including the
16209ff23fSmrg * next paragraph) shall be included in all copies or substantial
17209ff23fSmrg * portions of the Software.
18209ff23fSmrg *
19209ff23fSmrg * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
20209ff23fSmrg * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
21209ff23fSmrg * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
22209ff23fSmrg * NON-INFRINGEMENT.  IN NO EVENT SHALL ATI, VA LINUX SYSTEMS AND/OR
23209ff23fSmrg * THEIR SUPPLIERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
24209ff23fSmrg * WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
25209ff23fSmrg * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
26209ff23fSmrg * DEALINGS IN THE SOFTWARE.
27209ff23fSmrg */
28209ff23fSmrg
29209ff23fSmrg#ifdef HAVE_CONFIG_H
30209ff23fSmrg#include "config.h"
31209ff23fSmrg#endif
32209ff23fSmrg
33209ff23fSmrg#include <string.h>
34209ff23fSmrg#include <stdio.h>
35921a55d8Smrg#include <assert.h>
36921a55d8Smrg#include <math.h>
37209ff23fSmrg
38209ff23fSmrg/* X and server generic header files */
39209ff23fSmrg#include "xf86.h"
40209ff23fSmrg#include "xf86_OSproc.h"
41209ff23fSmrg#include "vgaHW.h"
42209ff23fSmrg#include "xf86Modes.h"
43209ff23fSmrg
44209ff23fSmrg/* Driver data structures */
45209ff23fSmrg#include "radeon.h"
46209ff23fSmrg#include "radeon_reg.h"
47209ff23fSmrg#include "radeon_macros.h"
48209ff23fSmrg#include "radeon_probe.h"
49209ff23fSmrg#include "radeon_version.h"
50209ff23fSmrg
51209ff23fSmrg#ifdef XF86DRI
52209ff23fSmrg#define _XF86DRI_SERVER_
53b7e1c893Smrg#include "radeon_drm.h"
54209ff23fSmrg#include "sarea.h"
55209ff23fSmrg#endif
56209ff23fSmrg
57209ff23fSmrgextern void atombios_crtc_mode_set(xf86CrtcPtr crtc,
58209ff23fSmrg				   DisplayModePtr mode,
59209ff23fSmrg				   DisplayModePtr adjusted_mode,
60209ff23fSmrg				   int x, int y);
61209ff23fSmrgextern void atombios_crtc_dpms(xf86CrtcPtr crtc, int mode);
62b7e1c893Smrgextern void
63b7e1c893SmrgRADEONInitDispBandwidthLegacy(ScrnInfoPtr pScrn,
64b7e1c893Smrg			      DisplayModePtr mode1, int pixel_bytes1,
65b7e1c893Smrg			      DisplayModePtr mode2, int pixel_bytes2);
66b7e1c893Smrgextern void
67b7e1c893SmrgRADEONInitDispBandwidthAVIVO(ScrnInfoPtr pScrn,
68b7e1c893Smrg			     DisplayModePtr mode1, int pixel_bytes1,
69b7e1c893Smrg			     DisplayModePtr mode2, int pixel_bytes2);
70209ff23fSmrg
71209ff23fSmrgvoid
7240732134Srjsradeon_do_crtc_dpms(xf86CrtcPtr crtc, int mode)
73209ff23fSmrg{
74209ff23fSmrg    RADEONInfoPtr info = RADEONPTR(crtc->scrn);
75209ff23fSmrg    RADEONEntPtr pRADEONEnt = RADEONEntPriv(crtc->scrn);
76209ff23fSmrg    xf86CrtcPtr crtc0 = pRADEONEnt->pCrtc[0];
7740732134Srjs    RADEONCrtcPrivatePtr radeon_crtc = crtc->driver_private;
78c503f109Smrg
79b7e1c893Smrg    if (IS_AVIVO_VARIANT || info->r4xx_atom) {
80209ff23fSmrg	atombios_crtc_dpms(crtc, mode);
81209ff23fSmrg    } else {
82209ff23fSmrg
83209ff23fSmrg	/* need to restore crtc1 before crtc0 or we may get a blank screen
84209ff23fSmrg	 * in some cases
85209ff23fSmrg	 */
86209ff23fSmrg	if ((radeon_crtc->crtc_id == 1) && (mode == DPMSModeOn)) {
87209ff23fSmrg	    if (crtc0->enabled)
88209ff23fSmrg		legacy_crtc_dpms(crtc0,  DPMSModeOff);
89209ff23fSmrg	}
90209ff23fSmrg
91209ff23fSmrg	legacy_crtc_dpms(crtc, mode);
92209ff23fSmrg
93209ff23fSmrg	if ((radeon_crtc->crtc_id == 1) && (mode == DPMSModeOn)) {
94209ff23fSmrg	    if (crtc0->enabled)
95209ff23fSmrg		legacy_crtc_dpms(crtc0, mode);
96209ff23fSmrg	}
97209ff23fSmrg    }
9840732134Srjs}
9940732134Srjs
10040732134Srjsvoid
10140732134Srjsradeon_crtc_dpms(xf86CrtcPtr crtc, int mode)
10240732134Srjs{
10340732134Srjs    RADEONCrtcPrivatePtr radeon_crtc = crtc->driver_private;
10440732134Srjs
10540732134Srjs    if ((mode == DPMSModeOn) && radeon_crtc->enabled)
10640732134Srjs	return;
10740732134Srjs
10840732134Srjs    if (mode == DPMSModeOff)
10940732134Srjs	radeon_crtc_modeset_ioctl(crtc, FALSE);
11040732134Srjs
11140732134Srjs    radeon_do_crtc_dpms(crtc, mode);
112209ff23fSmrg
113c503f109Smrg    if (mode != DPMSModeOff) {
114c503f109Smrg	radeon_crtc_modeset_ioctl(crtc, TRUE);
115c503f109Smrg	radeon_crtc_load_lut(crtc);
116c503f109Smrg    }
117c503f109Smrg
118209ff23fSmrg    if (mode == DPMSModeOn)
119209ff23fSmrg	radeon_crtc->enabled = TRUE;
120209ff23fSmrg    else
121209ff23fSmrg	radeon_crtc->enabled = FALSE;
122209ff23fSmrg}
123209ff23fSmrg
124209ff23fSmrgstatic Bool
125209ff23fSmrgradeon_crtc_mode_fixup(xf86CrtcPtr crtc, DisplayModePtr mode,
126209ff23fSmrg		     DisplayModePtr adjusted_mode)
127209ff23fSmrg{
128209ff23fSmrg    return TRUE;
129209ff23fSmrg}
130209ff23fSmrg
131209ff23fSmrgstatic void
132209ff23fSmrgradeon_crtc_mode_prepare(xf86CrtcPtr crtc)
133209ff23fSmrg{
134209ff23fSmrg    RADEONCrtcPrivatePtr radeon_crtc = crtc->driver_private;
135209ff23fSmrg
136209ff23fSmrg    if (radeon_crtc->enabled)
137209ff23fSmrg	crtc->funcs->hide_cursor(crtc);
138209ff23fSmrg}
139209ff23fSmrg
140209ff23fSmrgstatic uint32_t RADEONDiv(CARD64 n, uint32_t d)
141209ff23fSmrg{
142209ff23fSmrg    return (n + (d / 2)) / d;
143209ff23fSmrg}
144209ff23fSmrg
145ad43ddacSmrgstatic void
146ad43ddacSmrgRADEONComputePLL_old(RADEONPLLPtr pll,
147ad43ddacSmrg		     unsigned long freq,
148ad43ddacSmrg		     uint32_t *chosen_dot_clock_freq,
149ad43ddacSmrg		     uint32_t *chosen_feedback_div,
150ad43ddacSmrg		     uint32_t *chosen_frac_feedback_div,
151ad43ddacSmrg		     uint32_t *chosen_reference_div,
152ad43ddacSmrg		     uint32_t *chosen_post_div,
153ad43ddacSmrg		     int flags)
154209ff23fSmrg{
155209ff23fSmrg    uint32_t min_ref_div = pll->min_ref_div;
156209ff23fSmrg    uint32_t max_ref_div = pll->max_ref_div;
157ad43ddacSmrg    uint32_t min_post_div = pll->min_post_div;
158ad43ddacSmrg    uint32_t max_post_div = pll->max_post_div;
159ad43ddacSmrg    uint32_t min_fractional_feed_div = 0;
160ad43ddacSmrg    uint32_t max_fractional_feed_div = 0;
161209ff23fSmrg    uint32_t best_vco = pll->best_vco;
162209ff23fSmrg    uint32_t best_post_div = 1;
163209ff23fSmrg    uint32_t best_ref_div = 1;
164209ff23fSmrg    uint32_t best_feedback_div = 1;
165ad43ddacSmrg    uint32_t best_frac_feedback_div = 0;
166209ff23fSmrg    uint32_t best_freq = -1;
167209ff23fSmrg    uint32_t best_error = 0xffffffff;
168209ff23fSmrg    uint32_t best_vco_diff = 1;
169209ff23fSmrg    uint32_t post_div;
170209ff23fSmrg
171209ff23fSmrg    freq = freq * 1000;
172209ff23fSmrg
173042789b0Sveego/*    ErrorF("freq: %lu\n", freq); */
174209ff23fSmrg
175209ff23fSmrg    if (flags & RADEON_PLL_USE_REF_DIV)
176209ff23fSmrg	min_ref_div = max_ref_div = pll->reference_div;
177209ff23fSmrg    else {
178209ff23fSmrg	while (min_ref_div < max_ref_div-1) {
179209ff23fSmrg	    uint32_t mid=(min_ref_div+max_ref_div)/2;
180209ff23fSmrg	    uint32_t pll_in = pll->reference_freq / mid;
181209ff23fSmrg	    if (pll_in < pll->pll_in_min)
182209ff23fSmrg		max_ref_div = mid;
183209ff23fSmrg	    else if (pll_in > pll->pll_in_max)
184209ff23fSmrg		min_ref_div = mid;
185209ff23fSmrg	    else break;
186209ff23fSmrg	}
187209ff23fSmrg    }
188209ff23fSmrg
189ad43ddacSmrg    if (flags & RADEON_PLL_USE_POST_DIV)
190ad43ddacSmrg	min_post_div = max_post_div = pll->post_div;
191ad43ddacSmrg
192ad43ddacSmrg    if (flags & RADEON_PLL_USE_FRAC_FB_DIV) {
193ad43ddacSmrg	min_fractional_feed_div = pll->min_frac_feedback_div;
194ad43ddacSmrg	max_fractional_feed_div = pll->max_frac_feedback_div;
195ad43ddacSmrg    }
196ad43ddacSmrg
197ad43ddacSmrg    for (post_div = min_post_div; post_div <= max_post_div; ++post_div) {
198209ff23fSmrg	uint32_t ref_div;
199209ff23fSmrg
200209ff23fSmrg	if ((flags & RADEON_PLL_NO_ODD_POST_DIV) && (post_div & 1))
201209ff23fSmrg	    continue;
202209ff23fSmrg
203209ff23fSmrg	/* legacy radeons only have a few post_divs */
204209ff23fSmrg	if (flags & RADEON_PLL_LEGACY) {
205209ff23fSmrg	    if ((post_div == 5) ||
206209ff23fSmrg		(post_div == 7) ||
207209ff23fSmrg		(post_div == 9) ||
208209ff23fSmrg		(post_div == 10) ||
209209ff23fSmrg		(post_div == 11))
210209ff23fSmrg		continue;
211209ff23fSmrg	}
212209ff23fSmrg
213209ff23fSmrg	for (ref_div = min_ref_div; ref_div <= max_ref_div; ++ref_div) {
214ad43ddacSmrg	    uint32_t feedback_div, current_freq = 0, error, vco_diff;
215209ff23fSmrg	    uint32_t pll_in = pll->reference_freq / ref_div;
216209ff23fSmrg	    uint32_t min_feed_div = pll->min_feedback_div;
217209ff23fSmrg	    uint32_t max_feed_div = pll->max_feedback_div+1;
218209ff23fSmrg
219209ff23fSmrg	    if (pll_in < pll->pll_in_min || pll_in > pll->pll_in_max)
220209ff23fSmrg		continue;
221209ff23fSmrg
222209ff23fSmrg	    while (min_feed_div < max_feed_div) {
223209ff23fSmrg		uint32_t vco;
224ad43ddacSmrg		uint32_t min_frac_feed_div = min_fractional_feed_div;
225ad43ddacSmrg		uint32_t max_frac_feed_div = max_fractional_feed_div+1;
226ad43ddacSmrg		uint32_t frac_feedback_div;
227ad43ddacSmrg		CARD64 tmp;
228209ff23fSmrg
229209ff23fSmrg		feedback_div = (min_feed_div+max_feed_div)/2;
230209ff23fSmrg
231ad43ddacSmrg		tmp = (CARD64)pll->reference_freq * feedback_div;
232ad43ddacSmrg		vco = RADEONDiv(tmp, ref_div);
233209ff23fSmrg
234209ff23fSmrg		if (vco < pll->pll_out_min) {
235209ff23fSmrg		    min_feed_div = feedback_div+1;
236209ff23fSmrg		    continue;
237209ff23fSmrg		} else if(vco > pll->pll_out_max) {
238209ff23fSmrg		    max_feed_div = feedback_div;
239209ff23fSmrg		    continue;
240209ff23fSmrg		}
241209ff23fSmrg
242ad43ddacSmrg		while (min_frac_feed_div < max_frac_feed_div) {
243ad43ddacSmrg		    frac_feedback_div = (min_frac_feed_div+max_frac_feed_div)/2;
244ad43ddacSmrg		    tmp = (CARD64)pll->reference_freq * 10000 * feedback_div;
245ad43ddacSmrg		    tmp += (CARD64)pll->reference_freq * 1000 * frac_feedback_div;
246ad43ddacSmrg		    current_freq = RADEONDiv(tmp, ref_div * post_div);
247ad43ddacSmrg
2487421d112Schristos#define RD_ABS(a, b) ((a) > (b) ? (a) - (b) : (b) - (a))
249ad43ddacSmrg		    if (flags & RADEON_PLL_PREFER_CLOSEST_LOWER) {
250ad43ddacSmrg			error = freq - current_freq;
251c4ae5be6Smrg			error = (int32_t)error < 0 ? 0xffffffff : error;
252ad43ddacSmrg		    } else
2537421d112Schristos			error = RD_ABS(current_freq, freq);
2547421d112Schristos		    vco_diff = RD_ABS(vco, best_vco);
255ad43ddacSmrg
256ad43ddacSmrg		    if ((best_vco == 0 && error < best_error) ||
257ad43ddacSmrg			(best_vco != 0 &&
258ad43ddacSmrg			 (error < best_error - 100 ||
2597421d112Schristos			  (RD_ABS(error, best_error) < 100 && vco_diff < best_vco_diff )))) {
260209ff23fSmrg			best_post_div = post_div;
261209ff23fSmrg			best_ref_div = ref_div;
262209ff23fSmrg			best_feedback_div = feedback_div;
263ad43ddacSmrg			best_frac_feedback_div = frac_feedback_div;
264209ff23fSmrg			best_freq = current_freq;
265209ff23fSmrg			best_error = error;
266209ff23fSmrg			best_vco_diff = vco_diff;
267ad43ddacSmrg		    } else if (current_freq == freq) {
268ad43ddacSmrg			if (best_freq == -1) {
269ad43ddacSmrg			    best_post_div = post_div;
270ad43ddacSmrg			    best_ref_div = ref_div;
271ad43ddacSmrg			    best_feedback_div = feedback_div;
272ad43ddacSmrg			    best_frac_feedback_div = frac_feedback_div;
273ad43ddacSmrg			    best_freq = current_freq;
274ad43ddacSmrg			    best_error = error;
275ad43ddacSmrg			    best_vco_diff = vco_diff;
276ad43ddacSmrg			} else if (((flags & RADEON_PLL_PREFER_LOW_REF_DIV) && (ref_div < best_ref_div)) ||
277ad43ddacSmrg				   ((flags & RADEON_PLL_PREFER_HIGH_REF_DIV) && (ref_div > best_ref_div)) ||
278ad43ddacSmrg				   ((flags & RADEON_PLL_PREFER_LOW_FB_DIV) && (feedback_div < best_feedback_div)) ||
279ad43ddacSmrg				   ((flags & RADEON_PLL_PREFER_HIGH_FB_DIV) && (feedback_div > best_feedback_div)) ||
280ad43ddacSmrg				   ((flags & RADEON_PLL_PREFER_LOW_POST_DIV) && (post_div < best_post_div)) ||
281ad43ddacSmrg				   ((flags & RADEON_PLL_PREFER_HIGH_POST_DIV) && (post_div > best_post_div))) {
282ad43ddacSmrg			    best_post_div = post_div;
283ad43ddacSmrg			    best_ref_div = ref_div;
284ad43ddacSmrg			    best_feedback_div = feedback_div;
285ad43ddacSmrg			    best_frac_feedback_div = frac_feedback_div;
286ad43ddacSmrg			    best_freq = current_freq;
287ad43ddacSmrg			    best_error = error;
288ad43ddacSmrg			    best_vco_diff = vco_diff;
289ad43ddacSmrg			}
290209ff23fSmrg		    }
291ad43ddacSmrg		    if (current_freq < freq)
292ad43ddacSmrg			min_frac_feed_div = frac_feedback_div+1;
293ad43ddacSmrg		    else
294ad43ddacSmrg			max_frac_feed_div = frac_feedback_div;
295209ff23fSmrg		}
296209ff23fSmrg		if (current_freq < freq)
297209ff23fSmrg		    min_feed_div = feedback_div+1;
298209ff23fSmrg		else
299209ff23fSmrg		    max_feed_div = feedback_div;
300209ff23fSmrg	    }
301209ff23fSmrg	}
302209ff23fSmrg    }
303209ff23fSmrg
304042789b0Sveego/*
305209ff23fSmrg    ErrorF("best_freq: %u\n", (unsigned int)best_freq);
306209ff23fSmrg    ErrorF("best_feedback_div: %u\n", (unsigned int)best_feedback_div);
307ad43ddacSmrg    ErrorF("best_frac_feedback_div: %u\n", (unsigned int)best_frac_feedback_div);
308209ff23fSmrg    ErrorF("best_ref_div: %u\n", (unsigned int)best_ref_div);
309209ff23fSmrg    ErrorF("best_post_div: %u\n", (unsigned int)best_post_div);
310042789b0Sveego*/
311209ff23fSmrg
312209ff23fSmrg    if (best_freq == -1)
313209ff23fSmrg	FatalError("Couldn't find valid PLL dividers\n");
314209ff23fSmrg    *chosen_dot_clock_freq = best_freq / 10000;
315209ff23fSmrg    *chosen_feedback_div = best_feedback_div;
316ad43ddacSmrg    *chosen_frac_feedback_div = best_frac_feedback_div;
317209ff23fSmrg    *chosen_reference_div = best_ref_div;
318209ff23fSmrg    *chosen_post_div = best_post_div;
319209ff23fSmrg
320209ff23fSmrg}
321209ff23fSmrg
322ad43ddacSmrgstatic Bool
323ad43ddacSmrgcalc_fb_div(RADEONPLLPtr pll,
324ad43ddacSmrg            unsigned long freq,
325ad43ddacSmrg            int flags,
326ad43ddacSmrg            int post_div,
327ad43ddacSmrg	    int ref_div,
328ad43ddacSmrg            int *fb_div,
329ad43ddacSmrg            int *fb_div_frac)
330ad43ddacSmrg{
331ad43ddacSmrg    float ffreq = freq / 10;
332ad43ddacSmrg    float vco_freq = ffreq * post_div;
333ad43ddacSmrg    float feedback_divider = vco_freq * ref_div / pll->reference_freq;
334ad43ddacSmrg
335ad43ddacSmrg    if (flags & RADEON_PLL_USE_FRAC_FB_DIV) {
336ad43ddacSmrg        feedback_divider = floor((feedback_divider * 10.0) + 0.5) * 0.1;
337ad43ddacSmrg
338ad43ddacSmrg	*fb_div = floor(feedback_divider);
339ad43ddacSmrg        *fb_div_frac = fmod(feedback_divider, 1.0) * 10.0;
340ad43ddacSmrg
341ad43ddacSmrg    } else {
342ad43ddacSmrg        *fb_div = floor(feedback_divider + 0.5);
343ad43ddacSmrg        *fb_div_frac = 0;
344ad43ddacSmrg    }
345ad43ddacSmrg    if ((*fb_div < pll->min_feedback_div) || (*fb_div > pll->max_feedback_div))
346ad43ddacSmrg        return FALSE;
347ad43ddacSmrg    else
348ad43ddacSmrg        return TRUE;
349ad43ddacSmrg}
350ad43ddacSmrg
351ad43ddacSmrgstatic Bool
352ad43ddacSmrgcalc_fb_ref_div(RADEONPLLPtr pll,
353ad43ddacSmrg                unsigned long freq,
354ad43ddacSmrg                int flags,
355ad43ddacSmrg                int post_div,
356ad43ddacSmrg                int *fb_div,
357ad43ddacSmrg                int *fb_div_frac,
358ad43ddacSmrg                int *ref_div)
359ad43ddacSmrg{
360ad43ddacSmrg    float ffreq = freq / 10;
361ad43ddacSmrg    float max_error = ffreq * 0.0025;
362ad43ddacSmrg    float vco, error, pll_out;
363ad43ddacSmrg
364ad43ddacSmrg    for ((*ref_div) = pll->min_ref_div; (*ref_div) < pll->max_ref_div; ++(*ref_div)) {
365ad43ddacSmrg        if (calc_fb_div(pll, freq, flags, post_div, (*ref_div), fb_div, fb_div_frac)) {
366ad43ddacSmrg            vco = pll->reference_freq * ((*fb_div) + ((*fb_div_frac) * 0.1)) / (*ref_div);
367ad43ddacSmrg
368ad43ddacSmrg            if ((vco < pll->pll_out_min) || (vco > pll->pll_out_max))
369ad43ddacSmrg                continue;
370ad43ddacSmrg
371ad43ddacSmrg            pll_out = vco / post_div;
372ad43ddacSmrg
373ad43ddacSmrg            error = pll_out - ffreq;
374ad43ddacSmrg            if ((fabs(error) <= max_error) && (error >= 0))
375ad43ddacSmrg                return TRUE;
376ad43ddacSmrg        }
377ad43ddacSmrg    }
378ad43ddacSmrg    return FALSE;
379ad43ddacSmrg}
380ad43ddacSmrg
381ad43ddacSmrgstatic void
382ad43ddacSmrgRADEONComputePLL_new(RADEONPLLPtr pll,
383ad43ddacSmrg		     unsigned long freq,
384ad43ddacSmrg		     uint32_t *chosen_dot_clock_freq,
385ad43ddacSmrg		     uint32_t *chosen_feedback_div,
386ad43ddacSmrg		     uint32_t *chosen_frac_feedback_div,
387ad43ddacSmrg		     uint32_t *chosen_reference_div,
388ad43ddacSmrg		     uint32_t *chosen_post_div,
389ad43ddacSmrg		     int flags)
390ad43ddacSmrg{
391ad43ddacSmrg    float ffreq = freq / 10;
392ad43ddacSmrg    float vco_frequency;
393ad43ddacSmrg    int fb_div = 0, fb_div_frac = 0, post_div = 0, ref_div = 0;
394ad43ddacSmrg    uint32_t best_freq = 0;
395ad43ddacSmrg
396ad43ddacSmrg    if (flags & RADEON_PLL_USE_POST_DIV) {
397ad43ddacSmrg        post_div = pll->post_div;
398ad43ddacSmrg        if ((post_div < pll->min_post_div) || (post_div > pll->max_post_div))
399ad43ddacSmrg            goto done;
400ad43ddacSmrg        vco_frequency = ffreq * post_div;
401ad43ddacSmrg        if ((vco_frequency < pll->pll_out_min) || (vco_frequency > pll->pll_out_max))
402ad43ddacSmrg            goto done;
403ad43ddacSmrg
404ad43ddacSmrg        if (flags & RADEON_PLL_USE_REF_DIV) {
405ad43ddacSmrg            ref_div = pll->reference_div;
406ad43ddacSmrg            if ((ref_div < pll->min_ref_div) || (ref_div > pll->max_ref_div))
407ad43ddacSmrg                goto done;
408ad43ddacSmrg            if (!calc_fb_div(pll, freq, flags, post_div, ref_div, &fb_div, &fb_div_frac))
409ad43ddacSmrg                goto done;
410ad43ddacSmrg        }
411ad43ddacSmrg    } else {
412ad43ddacSmrg	for (post_div = pll->max_post_div; post_div >= pll->min_post_div; --post_div) {
413ad43ddacSmrg	    if (flags & RADEON_PLL_LEGACY) {
414ad43ddacSmrg		if ((post_div == 5) ||
415ad43ddacSmrg		    (post_div == 7) ||
416ad43ddacSmrg		    (post_div == 9) ||
417ad43ddacSmrg		    (post_div == 10) ||
418ad43ddacSmrg		    (post_div == 11))
419ad43ddacSmrg		    continue;
420ad43ddacSmrg	    }
421ad43ddacSmrg	    if ((flags & RADEON_PLL_NO_ODD_POST_DIV) && (post_div & 1))
422ad43ddacSmrg		continue;
423ad43ddacSmrg
424ad43ddacSmrg	    vco_frequency = ffreq * post_div;
425ad43ddacSmrg	    if ((vco_frequency < pll->pll_out_min) || (vco_frequency > pll->pll_out_max))
426ad43ddacSmrg		continue;
427ad43ddacSmrg	    if (flags & RADEON_PLL_USE_REF_DIV) {
428ad43ddacSmrg		ref_div = pll->reference_div;
429ad43ddacSmrg		if ((ref_div < pll->min_ref_div) || (ref_div > pll->max_ref_div))
430ad43ddacSmrg		    goto done;
431ad43ddacSmrg		if (calc_fb_div(pll, freq, flags, post_div, ref_div, &fb_div, &fb_div_frac))
432ad43ddacSmrg		    break;
433ad43ddacSmrg	    } else {
434ad43ddacSmrg		if (calc_fb_ref_div(pll, freq, flags, post_div, &fb_div, &fb_div_frac, &ref_div))
435ad43ddacSmrg		    break;
436ad43ddacSmrg	    }
437ad43ddacSmrg	}
438ad43ddacSmrg    }
439ad43ddacSmrg
440ad43ddacSmrg    best_freq = pll->reference_freq * 10 * fb_div;
441ad43ddacSmrg    best_freq += pll->reference_freq * fb_div_frac;
442ad43ddacSmrg    best_freq = best_freq / (ref_div * post_div);
443ad43ddacSmrg
444042789b0Sveego/*
445ad43ddacSmrg    ErrorF("best_freq: %u\n", (unsigned int)best_freq);
446ad43ddacSmrg    ErrorF("best_feedback_div: %u\n", (unsigned int)fb_div);
447ad43ddacSmrg    ErrorF("best_frac_feedback_div: %u\n", (unsigned int)fb_div_frac);
448ad43ddacSmrg    ErrorF("best_ref_div: %u\n", (unsigned int)ref_div);
449ad43ddacSmrg    ErrorF("best_post_div: %u\n", (unsigned int)post_div);
450042789b0Sveego*/
451ad43ddacSmrg
452ad43ddacSmrgdone:
453ad43ddacSmrg    if (best_freq == 0)
454ad43ddacSmrg	FatalError("Couldn't find valid PLL dividers\n");
455ad43ddacSmrg
456ad43ddacSmrg    *chosen_dot_clock_freq = best_freq;
457ad43ddacSmrg    *chosen_feedback_div = fb_div;
458ad43ddacSmrg    *chosen_frac_feedback_div = fb_div_frac;
459ad43ddacSmrg    *chosen_reference_div = ref_div;
460ad43ddacSmrg    *chosen_post_div = post_div;
461ad43ddacSmrg
462ad43ddacSmrg}
463ad43ddacSmrg
464ad43ddacSmrgvoid
4652f39173dSmrgRADEONComputePLL(xf86CrtcPtr crtc,
466ad43ddacSmrg		 RADEONPLLPtr pll,
467ad43ddacSmrg		 unsigned long freq,
468ad43ddacSmrg		 uint32_t *chosen_dot_clock_freq,
469ad43ddacSmrg		 uint32_t *chosen_feedback_div,
470ad43ddacSmrg		 uint32_t *chosen_frac_feedback_div,
471ad43ddacSmrg		 uint32_t *chosen_reference_div,
472ad43ddacSmrg		 uint32_t *chosen_post_div,
473ad43ddacSmrg		 int flags)
474ad43ddacSmrg{
4752f39173dSmrg    RADEONCrtcPrivatePtr radeon_crtc = crtc->driver_private;
476ad43ddacSmrg
4772f39173dSmrg    switch (radeon_crtc->pll_algo) {
4782f39173dSmrg    case RADEON_PLL_OLD:
4792f39173dSmrg	RADEONComputePLL_old(pll, freq, chosen_dot_clock_freq,
4802f39173dSmrg			     chosen_feedback_div, chosen_frac_feedback_div,
4812f39173dSmrg			     chosen_reference_div, chosen_post_div, flags);
4822f39173dSmrg	break;
4832f39173dSmrg    case RADEON_PLL_NEW:
4842f39173dSmrg	/* disable frac fb dividers */
4852f39173dSmrg	flags &= ~RADEON_PLL_USE_FRAC_FB_DIV;
4862f39173dSmrg	RADEONComputePLL_new(pll, freq, chosen_dot_clock_freq,
4872f39173dSmrg			     chosen_feedback_div, chosen_frac_feedback_div,
4882f39173dSmrg			     chosen_reference_div, chosen_post_div, flags);
4892f39173dSmrg	break;
490ad43ddacSmrg    }
491ad43ddacSmrg}
492ad43ddacSmrg
493209ff23fSmrgstatic void
494209ff23fSmrgradeon_crtc_mode_set(xf86CrtcPtr crtc, DisplayModePtr mode,
495209ff23fSmrg		     DisplayModePtr adjusted_mode, int x, int y)
496209ff23fSmrg{
497209ff23fSmrg    ScrnInfoPtr pScrn = crtc->scrn;
498209ff23fSmrg    RADEONInfoPtr info = RADEONPTR(pScrn);
499209ff23fSmrg
500b7e1c893Smrg    if (IS_AVIVO_VARIANT || info->r4xx_atom) {
501209ff23fSmrg	atombios_crtc_mode_set(crtc, mode, adjusted_mode, x, y);
502209ff23fSmrg    } else {
503209ff23fSmrg	legacy_crtc_mode_set(crtc, mode, adjusted_mode, x, y);
504209ff23fSmrg    }
505209ff23fSmrg}
506209ff23fSmrg
507209ff23fSmrgstatic void
508209ff23fSmrgradeon_crtc_mode_commit(xf86CrtcPtr crtc)
509209ff23fSmrg{
510209ff23fSmrg    if (crtc->scrn->pScreen != NULL)
511861b9feeSmrg#ifdef HAVE_XF86_CURSOR_RESET_CURSOR
512861b9feeSmrg	xf86CursorResetCursor(crtc->scrn->pScreen);
513861b9feeSmrg#else
514209ff23fSmrg	xf86_reload_cursors(crtc->scrn->pScreen);
515861b9feeSmrg#endif
516209ff23fSmrg}
517209ff23fSmrg
518209ff23fSmrgvoid
519209ff23fSmrgradeon_crtc_load_lut(xf86CrtcPtr crtc)
520209ff23fSmrg{
521209ff23fSmrg    ScrnInfoPtr pScrn = crtc->scrn;
522209ff23fSmrg    RADEONCrtcPrivatePtr radeon_crtc = crtc->driver_private;
523209ff23fSmrg    RADEONInfoPtr info = RADEONPTR(pScrn);
524209ff23fSmrg    unsigned char *RADEONMMIO = info->MMIO;
525209ff23fSmrg    int i;
526209ff23fSmrg
527209ff23fSmrg    if (!crtc->enabled)
528209ff23fSmrg	return;
529209ff23fSmrg
53040732134Srjs    radeon_save_palette_on_demand(pScrn, radeon_crtc->crtc_id);
53140732134Srjs
532ad43ddacSmrg    if (IS_DCE4_VARIANT) {
533ad43ddacSmrg	OUTREG(EVERGREEN_DC_LUT_CONTROL + radeon_crtc->crtc_offset, 0);
534209ff23fSmrg
535ad43ddacSmrg	OUTREG(EVERGREEN_DC_LUT_BLACK_OFFSET_BLUE + radeon_crtc->crtc_offset, 0);
536ad43ddacSmrg	OUTREG(EVERGREEN_DC_LUT_BLACK_OFFSET_GREEN + radeon_crtc->crtc_offset, 0);
537ad43ddacSmrg	OUTREG(EVERGREEN_DC_LUT_BLACK_OFFSET_RED + radeon_crtc->crtc_offset, 0);
538209ff23fSmrg
539ad43ddacSmrg	OUTREG(EVERGREEN_DC_LUT_WHITE_OFFSET_BLUE + radeon_crtc->crtc_offset, 0x0000ffff);
540ad43ddacSmrg	OUTREG(EVERGREEN_DC_LUT_WHITE_OFFSET_GREEN + radeon_crtc->crtc_offset, 0x0000ffff);
541ad43ddacSmrg	OUTREG(EVERGREEN_DC_LUT_WHITE_OFFSET_RED + radeon_crtc->crtc_offset, 0x0000ffff);
542209ff23fSmrg
543ad43ddacSmrg	OUTREG(EVERGREEN_DC_LUT_RW_MODE + radeon_crtc->crtc_offset, 0);
544ad43ddacSmrg	OUTREG(EVERGREEN_DC_LUT_WRITE_EN_MASK + radeon_crtc->crtc_offset, 0x00000007);
545209ff23fSmrg
546ad43ddacSmrg	for (i = 0; i < 256; i++) {
547ad43ddacSmrg	    OUTREG(EVERGREEN_DC_LUT_RW_INDEX + radeon_crtc->crtc_offset, i);
548ad43ddacSmrg	    OUTREG(EVERGREEN_DC_LUT_30_COLOR + radeon_crtc->crtc_offset,
549ad43ddacSmrg		   (((radeon_crtc->lut_r[i]) << 20) |
550ad43ddacSmrg		    ((radeon_crtc->lut_g[i]) << 10) |
551ad43ddacSmrg		    (radeon_crtc->lut_b[i])));
552ad43ddacSmrg	}
553ad43ddacSmrg    } else {
554ad43ddacSmrg	if (IS_AVIVO_VARIANT) {
555ad43ddacSmrg	    OUTREG(AVIVO_DC_LUTA_CONTROL + radeon_crtc->crtc_offset, 0);
556209ff23fSmrg
557ad43ddacSmrg	    OUTREG(AVIVO_DC_LUTA_BLACK_OFFSET_BLUE + radeon_crtc->crtc_offset, 0);
558ad43ddacSmrg	    OUTREG(AVIVO_DC_LUTA_BLACK_OFFSET_GREEN + radeon_crtc->crtc_offset, 0);
559ad43ddacSmrg	    OUTREG(AVIVO_DC_LUTA_BLACK_OFFSET_RED + radeon_crtc->crtc_offset, 0);
560209ff23fSmrg
561ad43ddacSmrg	    OUTREG(AVIVO_DC_LUTA_WHITE_OFFSET_BLUE + radeon_crtc->crtc_offset, 0x0000ffff);
562ad43ddacSmrg	    OUTREG(AVIVO_DC_LUTA_WHITE_OFFSET_GREEN + radeon_crtc->crtc_offset, 0x0000ffff);
563ad43ddacSmrg	    OUTREG(AVIVO_DC_LUTA_WHITE_OFFSET_RED + radeon_crtc->crtc_offset, 0x0000ffff);
564ad43ddacSmrg	}
565ad43ddacSmrg
566ad43ddacSmrg	PAL_SELECT(radeon_crtc->crtc_id);
567ad43ddacSmrg
568ad43ddacSmrg	if (IS_AVIVO_VARIANT) {
569ad43ddacSmrg	    OUTREG(AVIVO_DC_LUT_RW_MODE, 0);
570ad43ddacSmrg	    OUTREG(AVIVO_DC_LUT_WRITE_EN_MASK, 0x0000003f);
571ad43ddacSmrg	}
572ad43ddacSmrg
573ad43ddacSmrg	for (i = 0; i < 256; i++) {
574ad43ddacSmrg	    OUTPAL(i, radeon_crtc->lut_r[i], radeon_crtc->lut_g[i], radeon_crtc->lut_b[i]);
575ad43ddacSmrg	}
576ad43ddacSmrg
577ad43ddacSmrg	if (IS_AVIVO_VARIANT)
578ad43ddacSmrg	    OUTREG(AVIVO_D1GRPH_LUT_SEL + radeon_crtc->crtc_offset, radeon_crtc->crtc_id);
579209ff23fSmrg    }
580209ff23fSmrg}
581209ff23fSmrg
582209ff23fSmrgstatic void
583209ff23fSmrgradeon_crtc_gamma_set(xf86CrtcPtr crtc, uint16_t *red, uint16_t *green,
584209ff23fSmrg		      uint16_t *blue, int size)
585209ff23fSmrg{
586209ff23fSmrg    RADEONCrtcPrivatePtr radeon_crtc = crtc->driver_private;
5872f39173dSmrg    int i;
588209ff23fSmrg
5892f39173dSmrg    for (i = 0; i < 256; i++) {
5902f39173dSmrg	radeon_crtc->lut_r[i] = red[i] >> 6;
5912f39173dSmrg	radeon_crtc->lut_g[i] = green[i] >> 6;
5922f39173dSmrg	radeon_crtc->lut_b[i] = blue[i] >> 6;
593209ff23fSmrg    }
594209ff23fSmrg
595209ff23fSmrg    radeon_crtc_load_lut(crtc);
596209ff23fSmrg}
597209ff23fSmrg
598209ff23fSmrgstatic Bool
599209ff23fSmrgradeon_crtc_lock(xf86CrtcPtr crtc)
600209ff23fSmrg{
601209ff23fSmrg    ScrnInfoPtr		pScrn = crtc->scrn;
602209ff23fSmrg    RADEONInfoPtr  info = RADEONPTR(pScrn);
603209ff23fSmrg
604209ff23fSmrg#ifdef XF86DRI
605b7e1c893Smrg    if (info->cp->CPStarted && pScrn->pScreen) {
606209ff23fSmrg	DRILock(pScrn->pScreen, 0);
607209ff23fSmrg	if (info->accelOn)
608209ff23fSmrg	    RADEON_SYNC(info, pScrn);
609209ff23fSmrg	return TRUE;
610209ff23fSmrg    }
611209ff23fSmrg#endif
612209ff23fSmrg    if (info->accelOn)
613209ff23fSmrg        RADEON_SYNC(info, pScrn);
614209ff23fSmrg
615209ff23fSmrg    return FALSE;
616209ff23fSmrg
617209ff23fSmrg}
618209ff23fSmrg
619209ff23fSmrgstatic void
620209ff23fSmrgradeon_crtc_unlock(xf86CrtcPtr crtc)
621209ff23fSmrg{
622209ff23fSmrg    ScrnInfoPtr		pScrn = crtc->scrn;
623209ff23fSmrg    RADEONInfoPtr  info = RADEONPTR(pScrn);
624209ff23fSmrg
625209ff23fSmrg#ifdef XF86DRI
626b7e1c893Smrg	if (info->cp->CPStarted && pScrn->pScreen) DRIUnlock(pScrn->pScreen);
627209ff23fSmrg#endif
628209ff23fSmrg
629209ff23fSmrg    if (info->accelOn)
630209ff23fSmrg        RADEON_SYNC(info, pScrn);
631209ff23fSmrg}
632209ff23fSmrg
633209ff23fSmrg/**
634209ff23fSmrg * Allocates memory for a locked-in-framebuffer shadow of the given
635209ff23fSmrg * width and height for this CRTC's rotated shadow framebuffer.
636209ff23fSmrg */
637209ff23fSmrg
638209ff23fSmrgstatic void *
639209ff23fSmrgradeon_crtc_shadow_allocate (xf86CrtcPtr crtc, int width, int height)
640209ff23fSmrg{
641209ff23fSmrg    ScrnInfoPtr pScrn = crtc->scrn;
642209ff23fSmrg    RADEONInfoPtr  info = RADEONPTR(pScrn);
643209ff23fSmrg    RADEONCrtcPrivatePtr radeon_crtc = crtc->driver_private;
644209ff23fSmrg    unsigned long rotate_pitch;
645209ff23fSmrg    unsigned long rotate_offset;
646ad43ddacSmrg    int size;
647209ff23fSmrg    int cpp = pScrn->bitsPerPixel / 8;
648209ff23fSmrg
649b7e1c893Smrg    /* No rotation without accel */
650b7e1c893Smrg    if (((info->ChipFamily >= CHIP_FAMILY_R600) && !info->directRenderingEnabled) ||
651b7e1c893Smrg	xf86ReturnOptValBool(info->Options, OPTION_NOACCEL, FALSE)) {
652b7e1c893Smrg	xf86DrvMsg(pScrn->scrnIndex, X_ERROR,
653b7e1c893Smrg		   "Acceleration required for rotation\n");
654b7e1c893Smrg	return NULL;
655b7e1c893Smrg    }
656b7e1c893Smrg
657209ff23fSmrg    rotate_pitch = pScrn->displayWidth * cpp;
658209ff23fSmrg    size = rotate_pitch * height;
659209ff23fSmrg
660209ff23fSmrg    /* We could get close to what we want here by just creating a pixmap like
661209ff23fSmrg     * normal, but we have to lock it down in framebuffer, and there is no
662209ff23fSmrg     * setter for offscreen area locking in EXA currently.  So, we just
663209ff23fSmrg     * allocate offscreen memory and fake up a pixmap header for it.
664209ff23fSmrg     */
665ad43ddacSmrg    rotate_offset = radeon_legacy_allocate_memory(pScrn, &radeon_crtc->crtc_rotate_mem,
666ad43ddacSmrg		    size, RADEON_GPU_PAGE_SIZE, RADEON_GEM_DOMAIN_VRAM);
667b7e1c893Smrg    if (rotate_offset == 0)
668b7e1c893Smrg	return NULL;
669209ff23fSmrg
670209ff23fSmrg    return info->FB + rotate_offset;
671209ff23fSmrg}
672b7e1c893Smrg
673209ff23fSmrg/**
674209ff23fSmrg * Creates a pixmap for this CRTC's rotated shadow framebuffer.
675209ff23fSmrg */
676209ff23fSmrgstatic PixmapPtr
677209ff23fSmrgradeon_crtc_shadow_create(xf86CrtcPtr crtc, void *data, int width, int height)
678209ff23fSmrg{
679209ff23fSmrg    ScrnInfoPtr pScrn = crtc->scrn;
680209ff23fSmrg    unsigned long rotate_pitch;
681209ff23fSmrg    PixmapPtr rotate_pixmap;
682209ff23fSmrg    int cpp = pScrn->bitsPerPixel / 8;
683209ff23fSmrg
684209ff23fSmrg    if (!data)
685209ff23fSmrg	data = radeon_crtc_shadow_allocate(crtc, width, height);
686209ff23fSmrg
687209ff23fSmrg    rotate_pitch = pScrn->displayWidth * cpp;
688209ff23fSmrg
689209ff23fSmrg    rotate_pixmap = GetScratchPixmapHeader(pScrn->pScreen,
690209ff23fSmrg					   width, height,
691209ff23fSmrg					   pScrn->depth,
692209ff23fSmrg					   pScrn->bitsPerPixel,
693209ff23fSmrg					   rotate_pitch,
694209ff23fSmrg					   data);
695209ff23fSmrg
696209ff23fSmrg    if (rotate_pixmap == NULL) {
697209ff23fSmrg	xf86DrvMsg(pScrn->scrnIndex, X_ERROR,
698209ff23fSmrg		   "Couldn't allocate shadow pixmap for rotated CRTC\n");
699209ff23fSmrg    }
700209ff23fSmrg
701209ff23fSmrg    return rotate_pixmap;
702209ff23fSmrg}
703209ff23fSmrg
704209ff23fSmrgstatic void
705209ff23fSmrgradeon_crtc_shadow_destroy(xf86CrtcPtr crtc, PixmapPtr rotate_pixmap, void *data)
706209ff23fSmrg{
707209ff23fSmrg    ScrnInfoPtr pScrn = crtc->scrn;
708209ff23fSmrg    RADEONCrtcPrivatePtr radeon_crtc = crtc->driver_private;
709209ff23fSmrg
710209ff23fSmrg    if (rotate_pixmap)
711209ff23fSmrg	FreeScratchPixmapHeader(rotate_pixmap);
712b7e1c893Smrg
713209ff23fSmrg    if (data) {
714b7e1c893Smrg	radeon_legacy_free_memory(pScrn, radeon_crtc->crtc_rotate_mem);
715b7e1c893Smrg	radeon_crtc->crtc_rotate_mem = NULL;
716b7e1c893Smrg    }
717b7e1c893Smrg
718b7e1c893Smrg}
719b7e1c893Smrg
720b7e1c893Smrg#if XF86_CRTC_VERSION >= 2
721b7e1c893Smrg#include "radeon_atombios.h"
722b7e1c893Smrg
723b7e1c893Smrgextern AtomBiosResult
724b7e1c893Smrgatombios_lock_crtc(atomBiosHandlePtr atomBIOS, int crtc, int lock);
725b7e1c893Smrgextern void
726b7e1c893SmrgRADEONInitCrtcBase(xf86CrtcPtr crtc, RADEONSavePtr save,
727b7e1c893Smrg		   int x, int y);
728b7e1c893Smrgextern void
729b7e1c893SmrgRADEONInitCrtc2Base(xf86CrtcPtr crtc, RADEONSavePtr save,
730b7e1c893Smrg		    int x, int y);
731b7e1c893Smrgextern void
732b7e1c893SmrgRADEONRestoreCrtcBase(ScrnInfoPtr pScrn,
733b7e1c893Smrg		      RADEONSavePtr restore);
734b7e1c893Smrgextern void
735b7e1c893SmrgRADEONRestoreCrtc2Base(ScrnInfoPtr pScrn,
736b7e1c893Smrg		       RADEONSavePtr restore);
737b7e1c893Smrg
738b7e1c893Smrgstatic void
739b7e1c893Smrgradeon_crtc_set_origin(xf86CrtcPtr crtc, int x, int y)
740b7e1c893Smrg{
741b7e1c893Smrg    ScrnInfoPtr pScrn = crtc->scrn;
742b7e1c893Smrg    RADEONCrtcPrivatePtr radeon_crtc = crtc->driver_private;
743b7e1c893Smrg    RADEONInfoPtr info = RADEONPTR(pScrn);
744b7e1c893Smrg    unsigned char *RADEONMMIO = info->MMIO;
745b7e1c893Smrg
746ad43ddacSmrg
747ad43ddacSmrg    if (IS_DCE4_VARIANT) {
748ad43ddacSmrg	x &= ~3;
749ad43ddacSmrg	y &= ~1;
750ad43ddacSmrg	atombios_lock_crtc(info->atomBIOS, radeon_crtc->crtc_id, 1);
751ad43ddacSmrg	OUTREG(EVERGREEN_VIEWPORT_START + radeon_crtc->crtc_offset, (x << 16) | y);
752ad43ddacSmrg	atombios_lock_crtc(info->atomBIOS, radeon_crtc->crtc_id, 0);
753ad43ddacSmrg    } else if (IS_AVIVO_VARIANT) {
754b7e1c893Smrg	x &= ~3;
755b7e1c893Smrg	y &= ~1;
756b7e1c893Smrg	atombios_lock_crtc(info->atomBIOS, radeon_crtc->crtc_id, 1);
757b7e1c893Smrg	OUTREG(AVIVO_D1MODE_VIEWPORT_START + radeon_crtc->crtc_offset, (x << 16) | y);
758b7e1c893Smrg	atombios_lock_crtc(info->atomBIOS, radeon_crtc->crtc_id, 0);
759b7e1c893Smrg    } else {
760b7e1c893Smrg	switch (radeon_crtc->crtc_id) {
761b7e1c893Smrg	case 0:
762b7e1c893Smrg	    RADEONInitCrtcBase(crtc, info->ModeReg, x, y);
763b7e1c893Smrg	    RADEONRestoreCrtcBase(pScrn, info->ModeReg);
764b7e1c893Smrg	    break;
765b7e1c893Smrg	case 1:
766b7e1c893Smrg	    RADEONInitCrtc2Base(crtc, info->ModeReg, x, y);
767b7e1c893Smrg	    RADEONRestoreCrtc2Base(pScrn, info->ModeReg);
768b7e1c893Smrg	    break;
769b7e1c893Smrg	default:
770b7e1c893Smrg	    break;
771209ff23fSmrg	}
772209ff23fSmrg    }
773209ff23fSmrg}
774b7e1c893Smrg#endif
775209ff23fSmrg
776b7e1c893Smrg
777b7e1c893Smrgstatic xf86CrtcFuncsRec radeon_crtc_funcs = {
778209ff23fSmrg    .dpms = radeon_crtc_dpms,
779209ff23fSmrg    .save = NULL, /* XXX */
780209ff23fSmrg    .restore = NULL, /* XXX */
781209ff23fSmrg    .mode_fixup = radeon_crtc_mode_fixup,
782209ff23fSmrg    .prepare = radeon_crtc_mode_prepare,
783209ff23fSmrg    .mode_set = radeon_crtc_mode_set,
784209ff23fSmrg    .commit = radeon_crtc_mode_commit,
785209ff23fSmrg    .gamma_set = radeon_crtc_gamma_set,
786209ff23fSmrg    .lock = radeon_crtc_lock,
787209ff23fSmrg    .unlock = radeon_crtc_unlock,
788209ff23fSmrg    .shadow_create = radeon_crtc_shadow_create,
789209ff23fSmrg    .shadow_allocate = radeon_crtc_shadow_allocate,
790209ff23fSmrg    .shadow_destroy = radeon_crtc_shadow_destroy,
791209ff23fSmrg    .set_cursor_colors = radeon_crtc_set_cursor_colors,
792209ff23fSmrg    .set_cursor_position = radeon_crtc_set_cursor_position,
793209ff23fSmrg    .show_cursor = radeon_crtc_show_cursor,
794209ff23fSmrg    .hide_cursor = radeon_crtc_hide_cursor,
795209ff23fSmrg    .load_cursor_argb = radeon_crtc_load_cursor_argb,
796209ff23fSmrg    .destroy = NULL, /* XXX */
797b7e1c893Smrg#if XF86_CRTC_VERSION >= 2
798b7e1c893Smrg    .set_origin = radeon_crtc_set_origin,
799b7e1c893Smrg#endif
800209ff23fSmrg};
801209ff23fSmrg
802b7e1c893Smrgvoid
803b7e1c893SmrgRADEONInitDispBandwidth(ScrnInfoPtr pScrn)
804b7e1c893Smrg{
805b7e1c893Smrg    RADEONInfoPtr info = RADEONPTR(pScrn);
806b7e1c893Smrg    xf86CrtcConfigPtr   xf86_config = XF86_CRTC_CONFIG_PTR(pScrn);
807b7e1c893Smrg    DisplayModePtr mode1 = NULL, mode2 = NULL;
808b7e1c893Smrg    int pixel_bytes1 = info->CurrentLayout.pixel_bytes;
809b7e1c893Smrg    int pixel_bytes2 = info->CurrentLayout.pixel_bytes;
810b7e1c893Smrg
811ad43ddacSmrg    /* XXX fix me */
812ad43ddacSmrg    if (IS_DCE4_VARIANT)
813ad43ddacSmrg	return;
814ad43ddacSmrg
815b7e1c893Smrg    if (xf86_config->num_crtc == 2) {
816b7e1c893Smrg	if (xf86_config->crtc[1]->enabled &&
817b7e1c893Smrg	    xf86_config->crtc[0]->enabled) {
818b7e1c893Smrg	    mode1 = &xf86_config->crtc[0]->mode;
819b7e1c893Smrg	    mode2 = &xf86_config->crtc[1]->mode;
820b7e1c893Smrg	} else if (xf86_config->crtc[0]->enabled) {
821b7e1c893Smrg	    mode1 = &xf86_config->crtc[0]->mode;
822b7e1c893Smrg	} else if (xf86_config->crtc[1]->enabled) {
823b7e1c893Smrg	    mode2 = &xf86_config->crtc[1]->mode;
824b7e1c893Smrg	} else
825b7e1c893Smrg	    return;
826b7e1c893Smrg    } else {
827b7e1c893Smrg	if (info->IsPrimary)
828b7e1c893Smrg	    mode1 = &xf86_config->crtc[0]->mode;
829b7e1c893Smrg	else if (info->IsSecondary)
830b7e1c893Smrg	    mode2 = &xf86_config->crtc[0]->mode;
831b7e1c893Smrg	else if (xf86_config->crtc[0]->enabled)
832b7e1c893Smrg	    mode1 = &xf86_config->crtc[0]->mode;
833b7e1c893Smrg	else
834b7e1c893Smrg	    return;
835b7e1c893Smrg    }
836b7e1c893Smrg
837b7e1c893Smrg    if (IS_AVIVO_VARIANT)
838b7e1c893Smrg	RADEONInitDispBandwidthAVIVO(pScrn, mode1, pixel_bytes1, mode2, pixel_bytes2);
839b7e1c893Smrg    else
840b7e1c893Smrg	RADEONInitDispBandwidthLegacy(pScrn, mode1, pixel_bytes1, mode2, pixel_bytes2);
841b7e1c893Smrg}
842b7e1c893Smrg
843209ff23fSmrgBool RADEONAllocateControllers(ScrnInfoPtr pScrn, int mask)
844209ff23fSmrg{
845209ff23fSmrg    RADEONEntPtr pRADEONEnt = RADEONEntPriv(pScrn);
846209ff23fSmrg    RADEONInfoPtr  info = RADEONPTR(pScrn);
847ad43ddacSmrg    int i;
848209ff23fSmrg
849b7e1c893Smrg    if (!xf86ReturnOptValBool(info->Options, OPTION_NOACCEL, FALSE)) {
850b7e1c893Smrg	radeon_crtc_funcs.shadow_create = radeon_crtc_shadow_create;
851b7e1c893Smrg	radeon_crtc_funcs.shadow_allocate = radeon_crtc_shadow_allocate;
852b7e1c893Smrg	radeon_crtc_funcs.shadow_destroy = radeon_crtc_shadow_destroy;
853b7e1c893Smrg    }
854b7e1c893Smrg
855209ff23fSmrg    if (mask & 1) {
856209ff23fSmrg	if (pRADEONEnt->Controller[0])
857209ff23fSmrg	    return TRUE;
858209ff23fSmrg
859209ff23fSmrg	pRADEONEnt->pCrtc[0] = xf86CrtcCreate(pScrn, &radeon_crtc_funcs);
860209ff23fSmrg	if (!pRADEONEnt->pCrtc[0])
861209ff23fSmrg	    return FALSE;
862209ff23fSmrg
863209ff23fSmrg	pRADEONEnt->Controller[0] = xnfcalloc(sizeof(RADEONCrtcPrivateRec), 1);
864209ff23fSmrg	if (!pRADEONEnt->Controller[0])
865209ff23fSmrg	    return FALSE;
866209ff23fSmrg
867209ff23fSmrg	pRADEONEnt->pCrtc[0]->driver_private = pRADEONEnt->Controller[0];
868209ff23fSmrg	pRADEONEnt->Controller[0]->crtc_id = 0;
869209ff23fSmrg	pRADEONEnt->Controller[0]->crtc_offset = 0;
870b7e1c893Smrg	pRADEONEnt->Controller[0]->initialized = FALSE;
871209ff23fSmrg	if (info->allowColorTiling)
872209ff23fSmrg	    pRADEONEnt->Controller[0]->can_tile = 1;
873209ff23fSmrg	else
874209ff23fSmrg	    pRADEONEnt->Controller[0]->can_tile = 0;
875ad43ddacSmrg	pRADEONEnt->Controller[0]->pll_id = -1;
876209ff23fSmrg    }
877209ff23fSmrg
878209ff23fSmrg    if (mask & 2) {
879209ff23fSmrg	if (!pRADEONEnt->HasCRTC2)
880209ff23fSmrg	    return TRUE;
881209ff23fSmrg
882209ff23fSmrg	pRADEONEnt->pCrtc[1] = xf86CrtcCreate(pScrn, &radeon_crtc_funcs);
883209ff23fSmrg	if (!pRADEONEnt->pCrtc[1])
884209ff23fSmrg	    return FALSE;
885209ff23fSmrg
886209ff23fSmrg	pRADEONEnt->Controller[1] = xnfcalloc(sizeof(RADEONCrtcPrivateRec), 1);
887209ff23fSmrg	if (!pRADEONEnt->Controller[1])
888209ff23fSmrg	    {
8892f39173dSmrg		free(pRADEONEnt->Controller[0]);
890209ff23fSmrg		return FALSE;
891209ff23fSmrg	    }
892209ff23fSmrg
893209ff23fSmrg	pRADEONEnt->pCrtc[1]->driver_private = pRADEONEnt->Controller[1];
894209ff23fSmrg	pRADEONEnt->Controller[1]->crtc_id = 1;
895ad43ddacSmrg	if (IS_DCE4_VARIANT)
896ad43ddacSmrg	    pRADEONEnt->Controller[1]->crtc_offset = EVERGREEN_CRTC1_REGISTER_OFFSET;
897ad43ddacSmrg	else
898ad43ddacSmrg	    pRADEONEnt->Controller[1]->crtc_offset = AVIVO_D2CRTC_H_TOTAL - AVIVO_D1CRTC_H_TOTAL;
899b7e1c893Smrg	pRADEONEnt->Controller[1]->initialized = FALSE;
900209ff23fSmrg	if (info->allowColorTiling)
901209ff23fSmrg	    pRADEONEnt->Controller[1]->can_tile = 1;
902209ff23fSmrg	else
903209ff23fSmrg	    pRADEONEnt->Controller[1]->can_tile = 0;
904ad43ddacSmrg	pRADEONEnt->Controller[1]->pll_id = -1;
905ad43ddacSmrg    }
906ad43ddacSmrg
907ad43ddacSmrg    /* 6 crtcs on DCE4 chips */
908921a55d8Smrg    if (IS_DCE4_VARIANT && ((mask & 3) == 3) && !IS_DCE41_VARIANT) {
909ad43ddacSmrg	for (i = 2; i < RADEON_MAX_CRTC; i++) {
910ad43ddacSmrg	    pRADEONEnt->pCrtc[i] = xf86CrtcCreate(pScrn, &radeon_crtc_funcs);
911ad43ddacSmrg	    if (!pRADEONEnt->pCrtc[i])
912ad43ddacSmrg		return FALSE;
913ad43ddacSmrg
914ad43ddacSmrg	    pRADEONEnt->Controller[i] = xnfcalloc(sizeof(RADEONCrtcPrivateRec), 1);
915ad43ddacSmrg	    if (!pRADEONEnt->Controller[i])
916ad43ddacSmrg	    {
9172f39173dSmrg		free(pRADEONEnt->Controller[i]);
918ad43ddacSmrg		return FALSE;
919ad43ddacSmrg	    }
920ad43ddacSmrg
921ad43ddacSmrg	    pRADEONEnt->pCrtc[i]->driver_private = pRADEONEnt->Controller[i];
922ad43ddacSmrg	    pRADEONEnt->Controller[i]->crtc_id = i;
923ad43ddacSmrg	    switch (i) {
924ad43ddacSmrg	    case 0:
925ad43ddacSmrg		pRADEONEnt->Controller[i]->crtc_offset = EVERGREEN_CRTC0_REGISTER_OFFSET;
926ad43ddacSmrg		break;
927ad43ddacSmrg	    case 1:
928ad43ddacSmrg		pRADEONEnt->Controller[i]->crtc_offset = EVERGREEN_CRTC1_REGISTER_OFFSET;
929ad43ddacSmrg		break;
930ad43ddacSmrg	    case 2:
931ad43ddacSmrg		pRADEONEnt->Controller[i]->crtc_offset = EVERGREEN_CRTC2_REGISTER_OFFSET;
932ad43ddacSmrg		break;
933ad43ddacSmrg	    case 3:
934ad43ddacSmrg		pRADEONEnt->Controller[i]->crtc_offset = EVERGREEN_CRTC3_REGISTER_OFFSET;
935ad43ddacSmrg		break;
936ad43ddacSmrg	    case 4:
937ad43ddacSmrg		pRADEONEnt->Controller[i]->crtc_offset = EVERGREEN_CRTC4_REGISTER_OFFSET;
938ad43ddacSmrg		break;
939ad43ddacSmrg	    case 5:
940ad43ddacSmrg		pRADEONEnt->Controller[i]->crtc_offset = EVERGREEN_CRTC5_REGISTER_OFFSET;
941ad43ddacSmrg		break;
942ad43ddacSmrg	    }
943ad43ddacSmrg	    pRADEONEnt->Controller[i]->initialized = FALSE;
944ad43ddacSmrg	    if (info->allowColorTiling)
945ad43ddacSmrg		pRADEONEnt->Controller[i]->can_tile = 1;
946ad43ddacSmrg	    else
947ad43ddacSmrg		pRADEONEnt->Controller[i]->can_tile = 0;
948ad43ddacSmrg	    pRADEONEnt->Controller[i]->pll_id = -1;
949ad43ddacSmrg	}
950209ff23fSmrg    }
951209ff23fSmrg
952209ff23fSmrg    return TRUE;
953209ff23fSmrg}
954209ff23fSmrg
955209ff23fSmrg/**
956209ff23fSmrg * In the current world order, there are lists of modes per output, which may
957209ff23fSmrg * or may not include the mode that was asked to be set by XFree86's mode
958209ff23fSmrg * selection.  Find the closest one, in the following preference order:
959209ff23fSmrg *
960209ff23fSmrg * - Equality
961209ff23fSmrg * - Closer in size to the requested mode, but no larger
962209ff23fSmrg * - Closer in refresh rate to the requested mode.
963209ff23fSmrg */
964209ff23fSmrgDisplayModePtr
965209ff23fSmrgRADEONCrtcFindClosestMode(xf86CrtcPtr crtc, DisplayModePtr pMode)
966209ff23fSmrg{
967209ff23fSmrg    ScrnInfoPtr	pScrn = crtc->scrn;
968209ff23fSmrg    xf86CrtcConfigPtr   xf86_config = XF86_CRTC_CONFIG_PTR(pScrn);
969209ff23fSmrg    DisplayModePtr pBest = NULL, pScan = NULL;
970209ff23fSmrg    int i;
971209ff23fSmrg
972209ff23fSmrg    /* Assume that there's only one output connected to the given CRTC. */
973209ff23fSmrg    for (i = 0; i < xf86_config->num_output; i++)
974209ff23fSmrg    {
975209ff23fSmrg	xf86OutputPtr  output = xf86_config->output[i];
976209ff23fSmrg	if (output->crtc == crtc && output->probed_modes != NULL)
977209ff23fSmrg	{
978209ff23fSmrg	    pScan = output->probed_modes;
979209ff23fSmrg	    break;
980209ff23fSmrg	}
981209ff23fSmrg    }
982209ff23fSmrg
983209ff23fSmrg    /* If the pipe doesn't have any detected modes, just let the system try to
984209ff23fSmrg     * spam the desired mode in.
985209ff23fSmrg     */
986209ff23fSmrg    if (pScan == NULL) {
987209ff23fSmrg	RADEONCrtcPrivatePtr  radeon_crtc = crtc->driver_private;
988209ff23fSmrg	xf86DrvMsg(pScrn->scrnIndex, X_WARNING,
989209ff23fSmrg		   "No crtc mode list for crtc %d,"
990209ff23fSmrg		   "continuing with desired mode\n", radeon_crtc->crtc_id);
991209ff23fSmrg	return pMode;
992209ff23fSmrg    }
993209ff23fSmrg
994209ff23fSmrg    for (; pScan != NULL; pScan = pScan->next) {
995209ff23fSmrg	assert(pScan->VRefresh != 0.0);
996209ff23fSmrg
997209ff23fSmrg	/* If there's an exact match, we're done. */
998209ff23fSmrg	if (xf86ModesEqual(pScan, pMode)) {
999209ff23fSmrg	    pBest = pMode;
1000209ff23fSmrg	    break;
1001209ff23fSmrg	}
1002209ff23fSmrg
1003209ff23fSmrg	/* Reject if it's larger than the desired mode. */
1004209ff23fSmrg	if (pScan->HDisplay > pMode->HDisplay ||
1005209ff23fSmrg	    pScan->VDisplay > pMode->VDisplay)
1006209ff23fSmrg	{
1007209ff23fSmrg	    continue;
1008209ff23fSmrg	}
1009209ff23fSmrg
1010209ff23fSmrg	if (pBest == NULL) {
1011209ff23fSmrg	    pBest = pScan;
1012209ff23fSmrg	    continue;
1013209ff23fSmrg	}
1014209ff23fSmrg
1015209ff23fSmrg	/* Find if it's closer to the right size than the current best
1016209ff23fSmrg	 * option.
1017209ff23fSmrg	 */
1018209ff23fSmrg	if ((pScan->HDisplay > pBest->HDisplay &&
1019209ff23fSmrg	     pScan->VDisplay >= pBest->VDisplay) ||
1020209ff23fSmrg	    (pScan->HDisplay >= pBest->HDisplay &&
1021209ff23fSmrg	     pScan->VDisplay > pBest->VDisplay))
1022209ff23fSmrg	{
1023209ff23fSmrg	    pBest = pScan;
1024209ff23fSmrg	    continue;
1025209ff23fSmrg	}
1026209ff23fSmrg
1027209ff23fSmrg	/* Find if it's still closer to the right refresh than the current
1028209ff23fSmrg	 * best resolution.
1029209ff23fSmrg	 */
1030209ff23fSmrg	if (pScan->HDisplay == pBest->HDisplay &&
1031209ff23fSmrg	    pScan->VDisplay == pBest->VDisplay &&
1032209ff23fSmrg	    (fabs(pScan->VRefresh - pMode->VRefresh) <
1033209ff23fSmrg	     fabs(pBest->VRefresh - pMode->VRefresh))) {
1034209ff23fSmrg	    pBest = pScan;
1035209ff23fSmrg	}
1036209ff23fSmrg    }
1037209ff23fSmrg
1038209ff23fSmrg    if (pBest == NULL) {
1039209ff23fSmrg	xf86DrvMsg(pScrn->scrnIndex, X_WARNING,
1040209ff23fSmrg		   "No suitable mode found to program for the pipe.\n"
1041209ff23fSmrg		   "	continuing with desired mode %dx%d@%.1f\n",
1042209ff23fSmrg		   pMode->HDisplay, pMode->VDisplay, pMode->VRefresh);
1043209ff23fSmrg    } else if (!xf86ModesEqual(pBest, pMode)) {
1044209ff23fSmrg      RADEONCrtcPrivatePtr  radeon_crtc = crtc->driver_private;
1045209ff23fSmrg      int		    crtc = radeon_crtc->crtc_id;
1046209ff23fSmrg      xf86DrvMsg(pScrn->scrnIndex, X_WARNING,
1047209ff23fSmrg		   "Choosing pipe %d's mode %dx%d@%.1f instead of xf86 "
1048209ff23fSmrg		   "mode %dx%d@%.1f\n", crtc,
1049209ff23fSmrg		   pBest->HDisplay, pBest->VDisplay, pBest->VRefresh,
1050209ff23fSmrg		   pMode->HDisplay, pMode->VDisplay, pMode->VRefresh);
1051209ff23fSmrg	pMode = pBest;
1052209ff23fSmrg    }
1053209ff23fSmrg    return pMode;
1054209ff23fSmrg}
1055209ff23fSmrg
1056209ff23fSmrgvoid
1057209ff23fSmrgRADEONBlank(ScrnInfoPtr pScrn)
1058209ff23fSmrg{
1059209ff23fSmrg    xf86CrtcConfigPtr   xf86_config = XF86_CRTC_CONFIG_PTR(pScrn);
1060209ff23fSmrg    xf86OutputPtr output;
1061209ff23fSmrg    xf86CrtcPtr crtc;
1062209ff23fSmrg    int o, c;
1063209ff23fSmrg
1064209ff23fSmrg    for (c = 0; c < xf86_config->num_crtc; c++) {
1065209ff23fSmrg	crtc = xf86_config->crtc[c];
1066209ff23fSmrg	for (o = 0; o < xf86_config->num_output; o++) {
1067209ff23fSmrg	    output = xf86_config->output[o];
1068209ff23fSmrg	    if (output->crtc != crtc)
1069209ff23fSmrg		continue;
1070209ff23fSmrg
1071209ff23fSmrg	    output->funcs->dpms(output, DPMSModeOff);
1072209ff23fSmrg	}
1073209ff23fSmrg	crtc->funcs->dpms(crtc, DPMSModeOff);
1074209ff23fSmrg    }
1075209ff23fSmrg}
1076209ff23fSmrg
1077209ff23fSmrgvoid
1078209ff23fSmrgRADEONUnblank(ScrnInfoPtr pScrn)
1079209ff23fSmrg{
1080209ff23fSmrg    xf86CrtcConfigPtr   xf86_config = XF86_CRTC_CONFIG_PTR(pScrn);
1081209ff23fSmrg    xf86OutputPtr output;
1082209ff23fSmrg    xf86CrtcPtr crtc;
1083209ff23fSmrg    int o, c;
1084209ff23fSmrg
1085209ff23fSmrg    for (c = 0; c < xf86_config->num_crtc; c++) {
1086209ff23fSmrg	crtc = xf86_config->crtc[c];
1087209ff23fSmrg	if(!crtc->enabled)
1088209ff23fSmrg		continue;
1089209ff23fSmrg	crtc->funcs->dpms(crtc, DPMSModeOn);
1090209ff23fSmrg	for (o = 0; o < xf86_config->num_output; o++) {
1091209ff23fSmrg	    output = xf86_config->output[o];
1092209ff23fSmrg	    if (output->crtc != crtc)
1093209ff23fSmrg		continue;
1094209ff23fSmrg
1095209ff23fSmrg	    output->funcs->dpms(output, DPMSModeOn);
1096209ff23fSmrg	}
1097209ff23fSmrg    }
1098209ff23fSmrg}
1099209ff23fSmrg
1100209ff23fSmrgBool
1101209ff23fSmrgRADEONSetTiling(ScrnInfoPtr pScrn)
1102209ff23fSmrg{
1103209ff23fSmrg    xf86CrtcConfigPtr   xf86_config = XF86_CRTC_CONFIG_PTR(pScrn);
1104209ff23fSmrg    RADEONInfoPtr info = RADEONPTR(pScrn);
1105209ff23fSmrg    RADEONCrtcPrivatePtr radeon_crtc;
1106209ff23fSmrg    xf86CrtcPtr crtc;
1107209ff23fSmrg    int c;
1108209ff23fSmrg    int can_tile = 1;
1109209ff23fSmrg    Bool changed = FALSE;
1110209ff23fSmrg
1111209ff23fSmrg    for (c = 0; c < xf86_config->num_crtc; c++) {
1112209ff23fSmrg	crtc = xf86_config->crtc[c];
1113209ff23fSmrg	radeon_crtc = crtc->driver_private;
1114209ff23fSmrg
1115209ff23fSmrg	if (crtc->enabled) {
1116209ff23fSmrg	    if (!radeon_crtc->can_tile)
1117209ff23fSmrg		can_tile = 0;
1118209ff23fSmrg	}
1119209ff23fSmrg    }
1120209ff23fSmrg
1121209ff23fSmrg    if (info->tilingEnabled != can_tile)
1122209ff23fSmrg	changed = TRUE;
1123209ff23fSmrg
1124209ff23fSmrg#ifdef XF86DRI
1125209ff23fSmrg    if (info->directRenderingEnabled && (info->tilingEnabled != can_tile)) {
1126b7e1c893Smrg	drm_radeon_sarea_t *pSAREAPriv;
1127209ff23fSmrg	if (RADEONDRISetParam(pScrn, RADEON_SETPARAM_SWITCH_TILING, (can_tile ? 1 : 0)) < 0)
1128209ff23fSmrg	    xf86DrvMsg(pScrn->scrnIndex, X_ERROR,
1129209ff23fSmrg		       "[drm] failed changing tiling status\n");
1130209ff23fSmrg	/* if this is called during ScreenInit() we don't have pScrn->pScreen yet */
113168105dcbSveego	pSAREAPriv = DRIGetSAREAPrivate(xf86ScrnToScreen(pScrn));
1132209ff23fSmrg	info->tilingEnabled = pSAREAPriv->tiling_enabled ? TRUE : FALSE;
1133209ff23fSmrg    }
1134209ff23fSmrg#endif
1135209ff23fSmrg
1136209ff23fSmrg    return changed;
1137209ff23fSmrg}
1138