diff options
Diffstat (limited to 'drivers/gpu/drm/tilcdc/tilcdc_crtc.c')
| -rw-r--r-- | drivers/gpu/drm/tilcdc/tilcdc_crtc.c | 39 | 
1 files changed, 24 insertions, 15 deletions
| diff --git a/drivers/gpu/drm/tilcdc/tilcdc_crtc.c b/drivers/gpu/drm/tilcdc/tilcdc_crtc.c index 30213708fc99..29890d704cb4 100644 --- a/drivers/gpu/drm/tilcdc/tilcdc_crtc.c +++ b/drivers/gpu/drm/tilcdc/tilcdc_crtc.c @@ -203,18 +203,19 @@ static void tilcdc_crtc_set_clk(struct drm_crtc *crtc)  	struct drm_device *dev = crtc->dev;  	struct tilcdc_drm_private *priv = dev->dev_private;  	struct tilcdc_crtc *tilcdc_crtc = to_tilcdc_crtc(crtc); -	unsigned long clk_rate, real_rate, req_rate; +	unsigned long clk_rate, real_pclk_rate, pclk_rate;  	unsigned int clkdiv;  	int ret;  	clkdiv = 2; /* first try using a standard divider of 2 */  	/* mode.clock is in KHz, set_rate wants parameter in Hz */ -	req_rate = crtc->mode.clock * 1000; +	pclk_rate = crtc->mode.clock * 1000; -	ret = clk_set_rate(priv->clk, req_rate * clkdiv); +	ret = clk_set_rate(priv->clk, pclk_rate * clkdiv);  	clk_rate = clk_get_rate(priv->clk); -	if (ret < 0 || tilcdc_pclk_diff(req_rate, clk_rate) > 5) { +	real_pclk_rate = clk_rate / clkdiv; +	if (ret < 0 || tilcdc_pclk_diff(pclk_rate, real_pclk_rate) > 5) {  		/*  		 * If we fail to set the clock rate (some architectures don't  		 * use the common clock framework yet and may not implement @@ -229,7 +230,7 @@ static void tilcdc_crtc_set_clk(struct drm_crtc *crtc)  			return;  		} -		clkdiv = DIV_ROUND_CLOSEST(clk_rate, req_rate); +		clkdiv = DIV_ROUND_CLOSEST(clk_rate, pclk_rate);  		/*  		 * Emit a warning if the real clock rate resulting from the @@ -238,12 +239,12 @@ static void tilcdc_crtc_set_clk(struct drm_crtc *crtc)  		 * 5% is an arbitrary value - LCDs are usually quite tolerant  		 * about pixel clock rates.  		 */ -		real_rate = clkdiv * req_rate; +		real_pclk_rate = clk_rate / clkdiv; -		if (tilcdc_pclk_diff(clk_rate, real_rate) > 5) { +		if (tilcdc_pclk_diff(pclk_rate, real_pclk_rate) > 5) {  			dev_warn(dev->dev, -				 "effective pixel clock rate (%luHz) differs from the calculated rate (%luHz)\n", -				 clk_rate, real_rate); +				 "effective pixel clock rate (%luHz) differs from the requested rate (%luHz)\n", +				 real_pclk_rate, pclk_rate);  		}  	} @@ -393,7 +394,7 @@ static void tilcdc_crtc_set_mode(struct drm_crtc *crtc)  			return;  		}  	} -	reg |= info->fdd < 12; +	reg |= info->fdd << 12;  	tilcdc_write(dev, LCDC_RASTER_CTRL_REG, reg);  	if (info->invert_pxl_clk) @@ -515,6 +516,15 @@ static void tilcdc_crtc_off(struct drm_crtc *crtc, bool shutdown)  	drm_crtc_vblank_off(crtc); +	spin_lock_irq(&crtc->dev->event_lock); + +	if (crtc->state->event) { +		drm_crtc_send_vblank_event(crtc, crtc->state->event); +		crtc->state->event = NULL; +	} + +	spin_unlock_irq(&crtc->dev->event_lock); +  	tilcdc_crtc_disable_irqs(dev);  	pm_runtime_put_sync(dev->dev); @@ -904,13 +914,12 @@ irqreturn_t tilcdc_crtc_irq(struct drm_crtc *crtc)  	tilcdc_clear_irqstatus(dev, stat);  	if (stat & LCDC_END_OF_FRAME0) { -		unsigned long flags;  		bool skip_event = false;  		ktime_t now;  		now = ktime_get(); -		spin_lock_irqsave(&tilcdc_crtc->irq_lock, flags); +		spin_lock(&tilcdc_crtc->irq_lock);  		tilcdc_crtc->last_vblank = now; @@ -920,21 +929,21 @@ irqreturn_t tilcdc_crtc_irq(struct drm_crtc *crtc)  			skip_event = true;  		} -		spin_unlock_irqrestore(&tilcdc_crtc->irq_lock, flags); +		spin_unlock(&tilcdc_crtc->irq_lock);  		drm_crtc_handle_vblank(crtc);  		if (!skip_event) {  			struct drm_pending_vblank_event *event; -			spin_lock_irqsave(&dev->event_lock, flags); +			spin_lock(&dev->event_lock);  			event = tilcdc_crtc->event;  			tilcdc_crtc->event = NULL;  			if (event)  				drm_crtc_send_vblank_event(crtc, event); -			spin_unlock_irqrestore(&dev->event_lock, flags); +			spin_unlock(&dev->event_lock);  		}  		if (tilcdc_crtc->frame_intact) | 
