public inbox for linux-kernel@vger.kernel.org
 help / color / mirror / Atom feed
From: Benjamin Herrenschmidt <benh@kernel.crashing.org>
To: Kronos <kronos@kronoz.cjb.net>
Cc: Linux Kernel list <linux-kernel@vger.kernel.org>
Subject: Re: [4KSTACK][2.6.6] Stack overflow in radeonfb
Date: Fri, 14 May 2004 08:55:02 +1000	[thread overview]
Message-ID: <1084488901.3021.116.camel@gaston> (raw)
In-Reply-To: <20040513145640.GA3430@dreamland.darkstar.lan>


> 
> int radeonfb_set_par(struct fb_info *info)
> {
>         struct radeonfb_info *rinfo = info->par;
>         struct fb_var_screeninfo *mode = &info->var;
>         struct radeon_regs newmode;
>         
> struct radeon_regs is huge: 2356 bytes
> Quick fix (I'll test ASAP):

Wow, this is evil indeed, I didn't expect that struct to be that big,
but well... I did add a bunch of stuff to it lately.

Your fix looks good, I'll give it a try later today.


> --- linux-2.6/drivers/video/aty/radeon_base.c~	2004-05-13 16:51:08.000000000 +0200
> +++ linux-2.6/drivers/video/aty/radeon_base.c	2004-05-13 16:55:09.000000000 +0200
> @@ -1397,7 +1397,7 @@
>  {
>  	struct radeonfb_info *rinfo = info->par;
>  	struct fb_var_screeninfo *mode = &info->var;
> -	struct radeon_regs newmode;
> +	struct radeon_regs *newmode;
>  	int hTotal, vTotal, hSyncStart, hSyncEnd,
>  	    hSyncPol, vSyncStart, vSyncEnd, vSyncPol, cSync;
>  	u8 hsync_adj_tab[] = {0, 0x12, 9, 9, 6, 5};
> @@ -1410,6 +1410,10 @@
>  	int primary_mon = PRIMARY_MONITOR(rinfo);
>  	int depth = var_to_depth(mode);
>  
> +	newmode = kmalloc(sizeof(*newmode), GFP_KERNEL);
> +	if (!newmode)
> +		return -ENOMEM;
> +
>  	/* We always want engine to be idle on a mode switch, even
>  	 * if we won't actually change the mode
>  	 */
> @@ -1449,9 +1453,9 @@
>  
>  		if (rinfo->panel_info.use_bios_dividers) {
>  			nopllcalc = 1;
> -			newmode.ppll_div_3 = rinfo->panel_info.fbk_divider |
> +			newmode->ppll_div_3 = rinfo->panel_info.fbk_divider |
>  				(rinfo->panel_info.post_divider << 16);
> -			newmode.ppll_ref_div = rinfo->panel_info.ref_divider;
> +			newmode->ppll_ref_div = rinfo->panel_info.ref_divider;
>  		}
>  	}
>  	dotClock = 1000000000 / pixClock;
> @@ -1489,38 +1493,38 @@
>  
>  	hsync_start = hSyncStart - 8 + hsync_fudge;
>  
> -	newmode.crtc_gen_cntl = CRTC_EXT_DISP_EN | CRTC_EN |
> +	newmode->crtc_gen_cntl = CRTC_EXT_DISP_EN | CRTC_EN |
>  				(format << 8);
>  
>  	/* Clear auto-center etc... */
> -	newmode.crtc_more_cntl = rinfo->init_state.crtc_more_cntl;
> -	newmode.crtc_more_cntl &= 0xfffffff0;
> +	newmode->crtc_more_cntl = rinfo->init_state.crtc_more_cntl;
> +	newmode->crtc_more_cntl &= 0xfffffff0;
>  	
>  	if ((primary_mon == MT_DFP) || (primary_mon == MT_LCD)) {
> -		newmode.crtc_ext_cntl = VGA_ATI_LINEAR | XCRT_CNT_EN;
> +		newmode->crtc_ext_cntl = VGA_ATI_LINEAR | XCRT_CNT_EN;
>  		if (mirror)
> -			newmode.crtc_ext_cntl |= CRTC_CRT_ON;
> +			newmode->crtc_ext_cntl |= CRTC_CRT_ON;
>  
> -		newmode.crtc_gen_cntl &= ~(CRTC_DBL_SCAN_EN |
> +		newmode->crtc_gen_cntl &= ~(CRTC_DBL_SCAN_EN |
>  					   CRTC_INTERLACE_EN);
>  	} else {
> -		newmode.crtc_ext_cntl = VGA_ATI_LINEAR | XCRT_CNT_EN |
> +		newmode->crtc_ext_cntl = VGA_ATI_LINEAR | XCRT_CNT_EN |
>  					CRTC_CRT_ON;
>  	}
>  
> -	newmode.dac_cntl = /* INREG(DAC_CNTL) | */ DAC_MASK_ALL | DAC_VGA_ADR_EN |
> +	newmode->dac_cntl = /* INREG(DAC_CNTL) | */ DAC_MASK_ALL | DAC_VGA_ADR_EN |
>  			   DAC_8BIT_EN;
>  
> -	newmode.crtc_h_total_disp = ((((hTotal / 8) - 1) & 0x3ff) |
> +	newmode->crtc_h_total_disp = ((((hTotal / 8) - 1) & 0x3ff) |
>  				     (((mode->xres / 8) - 1) << 16));
>  
> -	newmode.crtc_h_sync_strt_wid = ((hsync_start & 0x1fff) |
> +	newmode->crtc_h_sync_strt_wid = ((hsync_start & 0x1fff) |
>  					(hsync_wid << 16) | (h_sync_pol << 23));
>  
> -	newmode.crtc_v_total_disp = ((vTotal - 1) & 0xffff) |
> +	newmode->crtc_v_total_disp = ((vTotal - 1) & 0xffff) |
>  				    ((mode->yres - 1) << 16);
>  
> -	newmode.crtc_v_sync_strt_wid = (((vSyncStart - 1) & 0xfff) |
> +	newmode->crtc_v_sync_strt_wid = (((vSyncStart - 1) & 0xfff) |
>  					 (vsync_wid << 16) | (v_sync_pol  << 23));
>  
>  	if (!radeon_accel_disabled()) {
> @@ -1529,18 +1533,18 @@
>   				& ~(0x3f)) >> 6;
>  
>  		/* Then, re-multiply it to get the CRTC pitch */
> -		newmode.crtc_pitch = (rinfo->pitch << 3) / ((mode->bits_per_pixel + 1) / 8);
> +		newmode->crtc_pitch = (rinfo->pitch << 3) / ((mode->bits_per_pixel + 1) / 8);
>  	} else
> -		newmode.crtc_pitch = (mode->xres_virtual >> 3);
> +		newmode->crtc_pitch = (mode->xres_virtual >> 3);
>  
> -	newmode.crtc_pitch |= (newmode.crtc_pitch << 16);
> +	newmode->crtc_pitch |= (newmode->crtc_pitch << 16);
>  
>  	/*
>  	 * It looks like recent chips have a problem with SURFACE_CNTL,
>  	 * setting SURF_TRANSLATION_DIS completely disables the
>  	 * swapper as well, so we leave it unset now.
>  	 */
> -	newmode.surface_cntl = 0;
> +	newmode->surface_cntl = 0;
>  
>  #if defined(__BIG_ENDIAN)
>  
> @@ -1550,28 +1554,28 @@
>  	 */
>  	switch (mode->bits_per_pixel) {
>  		case 16:
> -			newmode.surface_cntl |= NONSURF_AP0_SWP_16BPP;
> -			newmode.surface_cntl |= NONSURF_AP1_SWP_16BPP;
> +			newmode->surface_cntl |= NONSURF_AP0_SWP_16BPP;
> +			newmode->surface_cntl |= NONSURF_AP1_SWP_16BPP;
>  			break;
>  		case 24:	
>  		case 32:
> -			newmode.surface_cntl |= NONSURF_AP0_SWP_32BPP;
> -			newmode.surface_cntl |= NONSURF_AP1_SWP_32BPP;
> +			newmode->surface_cntl |= NONSURF_AP0_SWP_32BPP;
> +			newmode->surface_cntl |= NONSURF_AP1_SWP_32BPP;
>  			break;
>  	}
>  #endif
>  
>  	/* Clear surface registers */
>  	for (i=0; i<8; i++) {
> -		newmode.surf_lower_bound[i] = 0;
> -		newmode.surf_upper_bound[i] = 0x1f;
> -		newmode.surf_info[i] = 0;
> +		newmode->surf_lower_bound[i] = 0;
> +		newmode->surf_upper_bound[i] = 0x1f;
> +		newmode->surf_info[i] = 0;
>  	}
>  
>  	RTRACE("h_total_disp = 0x%x\t   hsync_strt_wid = 0x%x\n",
> -		newmode.crtc_h_total_disp, newmode.crtc_h_sync_strt_wid);
> +		newmode->crtc_h_total_disp, newmode->crtc_h_sync_strt_wid);
>  	RTRACE("v_total_disp = 0x%x\t   vsync_strt_wid = 0x%x\n",
> -		newmode.crtc_v_total_disp, newmode.crtc_v_sync_strt_wid);
> +		newmode->crtc_v_total_disp, newmode->crtc_v_sync_strt_wid);
>  
>  	rinfo->bpp = mode->bits_per_pixel;
>  	rinfo->depth = depth;
> @@ -1580,9 +1584,9 @@
>  	RTRACE("freq = %lu\n", (unsigned long)freq);
>  
>  	if (!nopllcalc)
> -		radeon_calc_pll_regs(rinfo, &newmode, freq);
> +		radeon_calc_pll_regs(rinfo, newmode, freq);
>  
> -	newmode.vclk_ecp_cntl = rinfo->init_state.vclk_ecp_cntl;
> +	newmode->vclk_ecp_cntl = rinfo->init_state.vclk_ecp_cntl;
>  
>  	if ((primary_mon == MT_DFP) || (primary_mon == MT_LCD)) {
>  		unsigned int hRatio, vRatio;
> @@ -1592,35 +1596,35 @@
>  		if (mode->yres > rinfo->panel_info.yres)
>  			mode->yres = rinfo->panel_info.yres;
>  
> -		newmode.fp_horz_stretch = (((rinfo->panel_info.xres / 8) - 1)
> +		newmode->fp_horz_stretch = (((rinfo->panel_info.xres / 8) - 1)
>  					   << HORZ_PANEL_SHIFT);
> -		newmode.fp_vert_stretch = ((rinfo->panel_info.yres - 1)
> +		newmode->fp_vert_stretch = ((rinfo->panel_info.yres - 1)
>  					   << VERT_PANEL_SHIFT);
>  
>  		if (mode->xres != rinfo->panel_info.xres) {
>  			hRatio = round_div(mode->xres * HORZ_STRETCH_RATIO_MAX,
>  					   rinfo->panel_info.xres);
> -			newmode.fp_horz_stretch = (((((unsigned long)hRatio) & HORZ_STRETCH_RATIO_MASK)) |
> -						   (newmode.fp_horz_stretch &
> +			newmode->fp_horz_stretch = (((((unsigned long)hRatio) & HORZ_STRETCH_RATIO_MASK)) |
> +						   (newmode->fp_horz_stretch &
>  						    (HORZ_PANEL_SIZE | HORZ_FP_LOOP_STRETCH |
>  						     HORZ_AUTO_RATIO_INC)));
> -			newmode.fp_horz_stretch |= (HORZ_STRETCH_BLEND |
> +			newmode->fp_horz_stretch |= (HORZ_STRETCH_BLEND |
>  						    HORZ_STRETCH_ENABLE);
>  		}
> -		newmode.fp_horz_stretch &= ~HORZ_AUTO_RATIO;
> +		newmode->fp_horz_stretch &= ~HORZ_AUTO_RATIO;
>  
>  		if (mode->yres != rinfo->panel_info.yres) {
>  			vRatio = round_div(mode->yres * VERT_STRETCH_RATIO_MAX,
>  					   rinfo->panel_info.yres);
> -			newmode.fp_vert_stretch = (((((unsigned long)vRatio) & VERT_STRETCH_RATIO_MASK)) |
> -						   (newmode.fp_vert_stretch &
> +			newmode->fp_vert_stretch = (((((unsigned long)vRatio) & VERT_STRETCH_RATIO_MASK)) |
> +						   (newmode->fp_vert_stretch &
>  						   (VERT_PANEL_SIZE | VERT_STRETCH_RESERVED)));
> -			newmode.fp_vert_stretch |= (VERT_STRETCH_BLEND |
> +			newmode->fp_vert_stretch |= (VERT_STRETCH_BLEND |
>  						    VERT_STRETCH_ENABLE);
>  		}
> -		newmode.fp_vert_stretch &= ~VERT_AUTO_RATIO_EN;
> +		newmode->fp_vert_stretch &= ~VERT_AUTO_RATIO_EN;
>  
> -		newmode.fp_gen_cntl = (rinfo->init_state.fp_gen_cntl & (u32)
> +		newmode->fp_gen_cntl = (rinfo->init_state.fp_gen_cntl & (u32)
>  				       ~(FP_SEL_CRTC2 |
>  					 FP_RMX_HVSYNC_CONTROL_EN |
>  					 FP_DFP_SYNC_SEL |
> @@ -1630,46 +1634,46 @@
>  					 FP_CRTC_USE_SHADOW_VEND |
>  					 FP_CRT_SYNC_ALT));
>  
> -		newmode.fp_gen_cntl |= (FP_CRTC_DONT_SHADOW_VPAR |
> +		newmode->fp_gen_cntl |= (FP_CRTC_DONT_SHADOW_VPAR |
>  					FP_CRTC_DONT_SHADOW_HEND);
>  
> -		newmode.lvds_gen_cntl = rinfo->init_state.lvds_gen_cntl;
> -		newmode.lvds_pll_cntl = rinfo->init_state.lvds_pll_cntl;
> -		newmode.tmds_crc = rinfo->init_state.tmds_crc;
> -		newmode.tmds_transmitter_cntl = rinfo->init_state.tmds_transmitter_cntl;
> +		newmode->lvds_gen_cntl = rinfo->init_state.lvds_gen_cntl;
> +		newmode->lvds_pll_cntl = rinfo->init_state.lvds_pll_cntl;
> +		newmode->tmds_crc = rinfo->init_state.tmds_crc;
> +		newmode->tmds_transmitter_cntl = rinfo->init_state.tmds_transmitter_cntl;
>  
>  		if (primary_mon == MT_LCD) {
> -			newmode.lvds_gen_cntl |= (LVDS_ON | LVDS_BLON);
> -			newmode.fp_gen_cntl &= ~(FP_FPON | FP_TMDS_EN);
> +			newmode->lvds_gen_cntl |= (LVDS_ON | LVDS_BLON);
> +			newmode->fp_gen_cntl &= ~(FP_FPON | FP_TMDS_EN);
>  		} else {
>  			/* DFP */
> -			newmode.fp_gen_cntl |= (FP_FPON | FP_TMDS_EN);
> -			newmode.tmds_transmitter_cntl = (TMDS_RAN_PAT_RST | TMDS_ICHCSEL) &
> +			newmode->fp_gen_cntl |= (FP_FPON | FP_TMDS_EN);
> +			newmode->tmds_transmitter_cntl = (TMDS_RAN_PAT_RST | TMDS_ICHCSEL) &
>  							 ~(TMDS_PLLRST);
>  			/* TMDS_PLL_EN bit is reversed on RV (and mobility) chips */
>  			if ((rinfo->family == CHIP_FAMILY_R300) ||
>  			    (rinfo->family == CHIP_FAMILY_R350) ||
>  			    (rinfo->family == CHIP_FAMILY_RV350) ||
>  			    (rinfo->family == CHIP_FAMILY_R200) || !rinfo->has_CRTC2)
> -				newmode.tmds_transmitter_cntl &= ~TMDS_PLL_EN;
> +				newmode->tmds_transmitter_cntl &= ~TMDS_PLL_EN;
>  			else
> -				newmode.tmds_transmitter_cntl |= TMDS_PLL_EN;
> -			newmode.crtc_ext_cntl &= ~CRTC_CRT_ON;
> +				newmode->tmds_transmitter_cntl |= TMDS_PLL_EN;
> +			newmode->crtc_ext_cntl &= ~CRTC_CRT_ON;
>  		}
>  
> -		newmode.fp_crtc_h_total_disp = (((rinfo->panel_info.hblank / 8) & 0x3ff) |
> +		newmode->fp_crtc_h_total_disp = (((rinfo->panel_info.hblank / 8) & 0x3ff) |
>  				(((mode->xres / 8) - 1) << 16));
> -		newmode.fp_crtc_v_total_disp = (rinfo->panel_info.vblank & 0xffff) |
> +		newmode->fp_crtc_v_total_disp = (rinfo->panel_info.vblank & 0xffff) |
>  				((mode->yres - 1) << 16);
> -		newmode.fp_h_sync_strt_wid = ((rinfo->panel_info.hOver_plus & 0x1fff) |
> +		newmode->fp_h_sync_strt_wid = ((rinfo->panel_info.hOver_plus & 0x1fff) |
>  				(hsync_wid << 16) | (h_sync_pol << 23));
> -		newmode.fp_v_sync_strt_wid = ((rinfo->panel_info.vOver_plus & 0xfff) |
> +		newmode->fp_v_sync_strt_wid = ((rinfo->panel_info.vOver_plus & 0xfff) |
>  				(vsync_wid << 16) | (v_sync_pol  << 23));
>  	}
>  
>  	/* do it! */
>  	if (!rinfo->asleep) {
> -		radeon_write_mode (rinfo, &newmode);
> +		radeon_write_mode (rinfo, newmode);
>  		/* (re)initialize the engine */
>  		if (!radeon_accel_disabled())
>  			radeonfb_engine_init (rinfo);
> @@ -1689,6 +1693,7 @@
>  	btext_update_display(rinfo->fb_base_phys, mode->xres, mode->yres,
>  			     rinfo->depth, info->fix.line_length);
>  #endif
> +	kfree(newmode);
>  
>  	return 0;
>  }
> 
> 
> Luca
-- 
Benjamin Herrenschmidt <benh@kernel.crashing.org>


  parent reply	other threads:[~2004-05-13 22:58 UTC|newest]

Thread overview: 32+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2004-05-13 13:48 [4KSTACK][2.6.6] Stack overflow in radeonfb Kronos
2004-05-13 14:03 ` Kronos
2004-05-13 14:56 ` Kronos
2004-05-13 15:15   ` Jörn Engel
2004-05-13 15:36     ` Valdis.Kletnieks
2004-05-13 16:02       ` Jörn Engel
2004-05-13 22:56     ` Benjamin Herrenschmidt
2004-05-14 10:00       ` Jörn Engel
2004-05-13 22:55   ` Benjamin Herrenschmidt [this message]
2004-05-14  1:21     ` Andrew Morton
2004-05-14  3:26       ` Randy.Dunlap
2004-05-14  9:49       ` Arjan van de Ven
2004-05-14 11:47         ` Jörn Engel
2004-05-14 22:15           ` Andrew Morton
2004-05-14 22:56             ` Chris Wright
2004-05-14 23:18               ` Andrew Morton
2004-05-14 23:19                 ` Chris Wright
2004-05-14 23:48                   ` Andrew Morton
2004-05-17 10:53                     ` Jörn Engel
2004-05-15  7:19             ` Arjan van de Ven
2004-05-17 23:35             ` Matt Mackall
2004-05-17 23:59               ` Andrew Morton
2004-05-26 10:06                 ` Jörn Engel
2004-05-26 10:08                   ` Jörn Engel
2004-05-19 10:28               ` William Lee Irwin III
2004-05-19 12:01                 ` William Lee Irwin III
2004-05-26 10:17                   ` Jörn Engel
     [not found]               ` <20040518051745.GK2151@krispykreme>
     [not found]                 ` <20040518171136.GC28735@waste.org>
     [not found]                   ` <20040518171959.GQ2151@krispykreme>
     [not found]                     ` <20040518174734.GE28735@waste.org>
2004-05-26 10:14                       ` Jörn Engel
2004-05-14 16:41     ` Kronos
2004-05-14 21:48       ` Benjamin Herrenschmidt
2004-05-14 22:34         ` Andrew Morton
2004-05-14 22:36           ` Benjamin Herrenschmidt

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to=1084488901.3021.116.camel@gaston \
    --to=benh@kernel.crashing.org \
    --cc=kronos@kronoz.cjb.net \
    --cc=linux-kernel@vger.kernel.org \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox