2014-12-09 00:09:10 +08:00
/*
* Copyright © 2014 Intel Corporation
*
* Permission is hereby granted , free of charge , to any person obtaining a
* copy of this software and associated documentation files ( the " Software " ) ,
* to deal in the Software without restriction , including without limitation
* the rights to use , copy , modify , merge , publish , distribute , sublicense ,
* and / or sell copies of the Software , and to permit persons to whom the
* Software is furnished to do so , subject to the following conditions :
*
* The above copyright notice and this permission notice ( including the next
* paragraph ) shall be included in all copies or substantial portions of the
* Software .
*
* THE SOFTWARE IS PROVIDED " AS IS " , WITHOUT WARRANTY OF ANY KIND , EXPRESS OR
* IMPLIED , INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY ,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT . IN NO EVENT SHALL
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM , DAMAGES OR OTHER
* LIABILITY , WHETHER IN AN ACTION OF CONTRACT , TORT OR OTHERWISE , ARISING
* FROM , OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
* DEALINGS IN THE SOFTWARE .
*/
2014-12-08 22:46:31 +08:00
/**
* DOC : Frame Buffer Compression ( FBC )
*
* FBC tries to save memory bandwidth ( and so power consumption ) by
* compressing the amount of memory used by the display . It is total
* transparent to user space and completely handled in the kernel .
2014-12-09 00:09:10 +08:00
*
* The benefits of FBC are mostly visible with solid backgrounds and
2014-12-08 22:46:31 +08:00
* variation - less patterns . It comes from keeping the memory footprint small
* and having fewer memory pages opened and accessed for refreshing the display .
2014-12-09 00:09:10 +08:00
*
2014-12-08 22:46:31 +08:00
* i915 is responsible to reserve stolen memory for FBC and configure its
* offset on proper registers . The hardware takes care of all
* compress / decompress . However there are many known cases where we have to
* forcibly disable it to allow proper screen updates .
2014-12-09 00:09:10 +08:00
*/
2014-12-08 22:46:31 +08:00
# include "intel_drv.h"
# include "i915_drv.h"
2014-12-09 00:09:10 +08:00
static void i8xx_fbc_disable ( struct drm_device * dev )
{
struct drm_i915_private * dev_priv = dev - > dev_private ;
u32 fbc_ctl ;
dev_priv - > fbc . enabled = false ;
/* Disable compression */
fbc_ctl = I915_READ ( FBC_CONTROL ) ;
if ( ( fbc_ctl & FBC_CTL_EN ) = = 0 )
return ;
fbc_ctl & = ~ FBC_CTL_EN ;
I915_WRITE ( FBC_CONTROL , fbc_ctl ) ;
/* Wait for compressing bit to clear */
if ( wait_for ( ( I915_READ ( FBC_STATUS ) & FBC_STAT_COMPRESSING ) = = 0 , 10 ) ) {
DRM_DEBUG_KMS ( " FBC idle timed out \n " ) ;
return ;
}
DRM_DEBUG_KMS ( " disabled FBC \n " ) ;
}
static void i8xx_fbc_enable ( struct drm_crtc * crtc )
{
struct drm_device * dev = crtc - > dev ;
struct drm_i915_private * dev_priv = dev - > dev_private ;
struct drm_framebuffer * fb = crtc - > primary - > fb ;
struct drm_i915_gem_object * obj = intel_fb_obj ( fb ) ;
struct intel_crtc * intel_crtc = to_intel_crtc ( crtc ) ;
int cfb_pitch ;
int i ;
u32 fbc_ctl ;
dev_priv - > fbc . enabled = true ;
2015-02-05 18:04:27 +08:00
/* Note: fbc.threshold == 1 for i8xx */
cfb_pitch = dev_priv - > fbc . uncompressed_size / FBC_LL_SIZE ;
2014-12-09 00:09:10 +08:00
if ( fb - > pitches [ 0 ] < cfb_pitch )
cfb_pitch = fb - > pitches [ 0 ] ;
/* FBC_CTL wants 32B or 64B units */
if ( IS_GEN2 ( dev ) )
cfb_pitch = ( cfb_pitch / 32 ) - 1 ;
else
cfb_pitch = ( cfb_pitch / 64 ) - 1 ;
/* Clear old tags */
for ( i = 0 ; i < ( FBC_LL_SIZE / 32 ) + 1 ; i + + )
I915_WRITE ( FBC_TAG + ( i * 4 ) , 0 ) ;
if ( IS_GEN4 ( dev ) ) {
u32 fbc_ctl2 ;
/* Set it up... */
fbc_ctl2 = FBC_CTL_FENCE_DBL | FBC_CTL_IDLE_IMM | FBC_CTL_CPU_FENCE ;
fbc_ctl2 | = FBC_CTL_PLANE ( intel_crtc - > plane ) ;
I915_WRITE ( FBC_CONTROL2 , fbc_ctl2 ) ;
I915_WRITE ( FBC_FENCE_OFF , crtc - > y ) ;
}
/* enable it... */
fbc_ctl = I915_READ ( FBC_CONTROL ) ;
fbc_ctl & = 0x3fff < < FBC_CTL_INTERVAL_SHIFT ;
fbc_ctl | = FBC_CTL_EN | FBC_CTL_PERIODIC ;
if ( IS_I945GM ( dev ) )
fbc_ctl | = FBC_CTL_C3_IDLE ; /* 945 needs special SR handling */
fbc_ctl | = ( cfb_pitch & 0xff ) < < FBC_CTL_STRIDE_SHIFT ;
fbc_ctl | = obj - > fence_reg ;
I915_WRITE ( FBC_CONTROL , fbc_ctl ) ;
DRM_DEBUG_KMS ( " enabled FBC, pitch %d, yoff %d, plane %c \n " ,
cfb_pitch , crtc - > y , plane_name ( intel_crtc - > plane ) ) ;
}
static bool i8xx_fbc_enabled ( struct drm_device * dev )
{
struct drm_i915_private * dev_priv = dev - > dev_private ;
return I915_READ ( FBC_CONTROL ) & FBC_CTL_EN ;
}
static void g4x_fbc_enable ( struct drm_crtc * crtc )
{
struct drm_device * dev = crtc - > dev ;
struct drm_i915_private * dev_priv = dev - > dev_private ;
struct drm_framebuffer * fb = crtc - > primary - > fb ;
struct drm_i915_gem_object * obj = intel_fb_obj ( fb ) ;
struct intel_crtc * intel_crtc = to_intel_crtc ( crtc ) ;
u32 dpfc_ctl ;
dev_priv - > fbc . enabled = true ;
dpfc_ctl = DPFC_CTL_PLANE ( intel_crtc - > plane ) | DPFC_SR_EN ;
if ( drm_format_plane_cpp ( fb - > pixel_format , 0 ) = = 2 )
dpfc_ctl | = DPFC_CTL_LIMIT_2X ;
else
dpfc_ctl | = DPFC_CTL_LIMIT_1X ;
dpfc_ctl | = DPFC_CTL_FENCE_EN | obj - > fence_reg ;
I915_WRITE ( DPFC_FENCE_YOFF , crtc - > y ) ;
/* enable it... */
I915_WRITE ( DPFC_CONTROL , dpfc_ctl | DPFC_CTL_EN ) ;
DRM_DEBUG_KMS ( " enabled fbc on plane %c \n " , plane_name ( intel_crtc - > plane ) ) ;
}
static void g4x_fbc_disable ( struct drm_device * dev )
{
struct drm_i915_private * dev_priv = dev - > dev_private ;
u32 dpfc_ctl ;
dev_priv - > fbc . enabled = false ;
/* Disable compression */
dpfc_ctl = I915_READ ( DPFC_CONTROL ) ;
if ( dpfc_ctl & DPFC_CTL_EN ) {
dpfc_ctl & = ~ DPFC_CTL_EN ;
I915_WRITE ( DPFC_CONTROL , dpfc_ctl ) ;
DRM_DEBUG_KMS ( " disabled FBC \n " ) ;
}
}
static bool g4x_fbc_enabled ( struct drm_device * dev )
{
struct drm_i915_private * dev_priv = dev - > dev_private ;
return I915_READ ( DPFC_CONTROL ) & DPFC_CTL_EN ;
}
2015-02-14 03:23:46 +08:00
static void intel_fbc_nuke ( struct drm_i915_private * dev_priv )
2014-12-09 00:09:10 +08:00
{
2015-02-14 03:23:46 +08:00
I915_WRITE ( MSG_FBC_REND_STATE , FBC_REND_NUKE ) ;
POSTING_READ ( MSG_FBC_REND_STATE ) ;
2014-12-09 00:09:10 +08:00
}
static void ilk_fbc_enable ( struct drm_crtc * crtc )
{
struct drm_device * dev = crtc - > dev ;
struct drm_i915_private * dev_priv = dev - > dev_private ;
struct drm_framebuffer * fb = crtc - > primary - > fb ;
struct drm_i915_gem_object * obj = intel_fb_obj ( fb ) ;
struct intel_crtc * intel_crtc = to_intel_crtc ( crtc ) ;
u32 dpfc_ctl ;
2015-06-30 21:53:05 +08:00
int threshold = dev_priv - > fbc . threshold ;
2014-12-09 00:09:10 +08:00
dev_priv - > fbc . enabled = true ;
dpfc_ctl = DPFC_CTL_PLANE ( intel_crtc - > plane ) ;
if ( drm_format_plane_cpp ( fb - > pixel_format , 0 ) = = 2 )
2015-06-30 21:53:05 +08:00
threshold + + ;
2014-12-09 00:09:10 +08:00
2015-06-30 21:53:05 +08:00
switch ( threshold ) {
2014-12-09 00:09:10 +08:00
case 4 :
case 3 :
dpfc_ctl | = DPFC_CTL_LIMIT_4X ;
break ;
case 2 :
dpfc_ctl | = DPFC_CTL_LIMIT_2X ;
break ;
case 1 :
dpfc_ctl | = DPFC_CTL_LIMIT_1X ;
break ;
}
dpfc_ctl | = DPFC_CTL_FENCE_EN ;
if ( IS_GEN5 ( dev ) )
dpfc_ctl | = obj - > fence_reg ;
I915_WRITE ( ILK_DPFC_FENCE_YOFF , crtc - > y ) ;
I915_WRITE ( ILK_FBC_RT_BASE , i915_gem_obj_ggtt_offset ( obj ) | ILK_FBC_RT_VALID ) ;
/* enable it... */
I915_WRITE ( ILK_DPFC_CONTROL , dpfc_ctl | DPFC_CTL_EN ) ;
if ( IS_GEN6 ( dev ) ) {
I915_WRITE ( SNB_DPFC_CTL_SA ,
SNB_CPU_FENCE_ENABLE | obj - > fence_reg ) ;
I915_WRITE ( DPFC_CPU_FENCE_OFFSET , crtc - > y ) ;
}
2015-02-14 03:23:46 +08:00
intel_fbc_nuke ( dev_priv ) ;
2014-12-09 00:09:10 +08:00
DRM_DEBUG_KMS ( " enabled fbc on plane %c \n " , plane_name ( intel_crtc - > plane ) ) ;
}
static void ilk_fbc_disable ( struct drm_device * dev )
{
struct drm_i915_private * dev_priv = dev - > dev_private ;
u32 dpfc_ctl ;
dev_priv - > fbc . enabled = false ;
/* Disable compression */
dpfc_ctl = I915_READ ( ILK_DPFC_CONTROL ) ;
if ( dpfc_ctl & DPFC_CTL_EN ) {
dpfc_ctl & = ~ DPFC_CTL_EN ;
I915_WRITE ( ILK_DPFC_CONTROL , dpfc_ctl ) ;
DRM_DEBUG_KMS ( " disabled FBC \n " ) ;
}
}
static bool ilk_fbc_enabled ( struct drm_device * dev )
{
struct drm_i915_private * dev_priv = dev - > dev_private ;
return I915_READ ( ILK_DPFC_CONTROL ) & DPFC_CTL_EN ;
}
static void gen7_fbc_enable ( struct drm_crtc * crtc )
{
struct drm_device * dev = crtc - > dev ;
struct drm_i915_private * dev_priv = dev - > dev_private ;
struct drm_framebuffer * fb = crtc - > primary - > fb ;
struct drm_i915_gem_object * obj = intel_fb_obj ( fb ) ;
struct intel_crtc * intel_crtc = to_intel_crtc ( crtc ) ;
u32 dpfc_ctl ;
2015-06-30 21:53:05 +08:00
int threshold = dev_priv - > fbc . threshold ;
2014-12-09 00:09:10 +08:00
dev_priv - > fbc . enabled = true ;
2015-06-13 01:36:21 +08:00
dpfc_ctl = 0 ;
if ( IS_IVYBRIDGE ( dev ) )
dpfc_ctl | = IVB_DPFC_CTL_PLANE ( intel_crtc - > plane ) ;
2014-12-09 00:09:10 +08:00
if ( drm_format_plane_cpp ( fb - > pixel_format , 0 ) = = 2 )
2015-06-30 21:53:05 +08:00
threshold + + ;
2014-12-09 00:09:10 +08:00
2015-06-30 21:53:05 +08:00
switch ( threshold ) {
2014-12-09 00:09:10 +08:00
case 4 :
case 3 :
dpfc_ctl | = DPFC_CTL_LIMIT_4X ;
break ;
case 2 :
dpfc_ctl | = DPFC_CTL_LIMIT_2X ;
break ;
case 1 :
dpfc_ctl | = DPFC_CTL_LIMIT_1X ;
break ;
}
dpfc_ctl | = IVB_DPFC_CTL_FENCE_EN ;
if ( dev_priv - > fbc . false_color )
dpfc_ctl | = FBC_CTL_FALSE_COLOR ;
I915_WRITE ( ILK_DPFC_CONTROL , dpfc_ctl | DPFC_CTL_EN ) ;
if ( IS_IVYBRIDGE ( dev ) ) {
/* WaFbcAsynchFlipDisableFbcQueue:ivb */
I915_WRITE ( ILK_DISPLAY_CHICKEN1 ,
I915_READ ( ILK_DISPLAY_CHICKEN1 ) |
ILK_FBCQ_DIS ) ;
} else {
/* WaFbcAsynchFlipDisableFbcQueue:hsw,bdw */
I915_WRITE ( CHICKEN_PIPESL_1 ( intel_crtc - > pipe ) ,
I915_READ ( CHICKEN_PIPESL_1 ( intel_crtc - > pipe ) ) |
HSW_FBCQ_DIS ) ;
}
I915_WRITE ( SNB_DPFC_CTL_SA ,
SNB_CPU_FENCE_ENABLE | obj - > fence_reg ) ;
I915_WRITE ( DPFC_CPU_FENCE_OFFSET , crtc - > y ) ;
2015-02-14 03:23:46 +08:00
intel_fbc_nuke ( dev_priv ) ;
2014-12-09 00:09:10 +08:00
DRM_DEBUG_KMS ( " enabled fbc on plane %c \n " , plane_name ( intel_crtc - > plane ) ) ;
}
2014-12-08 22:46:31 +08:00
/**
* intel_fbc_enabled - Is FBC enabled ?
* @ dev : the drm_device
*
* This function is used to verify the current state of FBC .
* FIXME : This should be tracked in the plane config eventually
* instead of queried at runtime for most callers .
*/
2014-12-09 00:09:10 +08:00
bool intel_fbc_enabled ( struct drm_device * dev )
{
struct drm_i915_private * dev_priv = dev - > dev_private ;
return dev_priv - > fbc . enabled ;
}
static void intel_fbc_work_fn ( struct work_struct * __work )
{
struct intel_fbc_work * work =
container_of ( to_delayed_work ( __work ) ,
struct intel_fbc_work , work ) ;
struct drm_device * dev = work - > crtc - > dev ;
struct drm_i915_private * dev_priv = dev - > dev_private ;
2015-07-03 06:25:10 +08:00
mutex_lock ( & dev_priv - > fbc . lock ) ;
2014-12-09 00:09:10 +08:00
if ( work = = dev_priv - > fbc . fbc_work ) {
/* Double check that we haven't switched fb without cancelling
* the prior work .
*/
if ( work - > crtc - > primary - > fb = = work - > fb ) {
2015-07-08 02:26:03 +08:00
dev_priv - > fbc . enable_fbc ( work - > crtc ) ;
2014-12-09 00:09:10 +08:00
2015-02-10 00:46:29 +08:00
dev_priv - > fbc . crtc = to_intel_crtc ( work - > crtc ) ;
2014-12-09 00:09:10 +08:00
dev_priv - > fbc . fb_id = work - > crtc - > primary - > fb - > base . id ;
dev_priv - > fbc . y = work - > crtc - > y ;
}
dev_priv - > fbc . fbc_work = NULL ;
}
2015-07-03 06:25:10 +08:00
mutex_unlock ( & dev_priv - > fbc . lock ) ;
2014-12-09 00:09:10 +08:00
kfree ( work ) ;
}
static void intel_fbc_cancel_work ( struct drm_i915_private * dev_priv )
{
2015-07-03 06:25:10 +08:00
WARN_ON ( ! mutex_is_locked ( & dev_priv - > fbc . lock ) ) ;
2014-12-09 00:09:10 +08:00
if ( dev_priv - > fbc . fbc_work = = NULL )
return ;
DRM_DEBUG_KMS ( " cancelling pending FBC enable \n " ) ;
/* Synchronisation is provided by struct_mutex and checking of
* dev_priv - > fbc . fbc_work , so we can perform the cancellation
* entirely asynchronously .
*/
if ( cancel_delayed_work ( & dev_priv - > fbc . fbc_work - > work ) )
/* tasklet was killed before being run, clean up */
kfree ( dev_priv - > fbc . fbc_work ) ;
/* Mark the work as no longer wanted so that if it does
* wake - up ( because the work was already running and waiting
* for our mutex ) , it will discover that is no longer
* necessary to run .
*/
dev_priv - > fbc . fbc_work = NULL ;
}
static void intel_fbc_enable ( struct drm_crtc * crtc )
{
struct intel_fbc_work * work ;
struct drm_device * dev = crtc - > dev ;
struct drm_i915_private * dev_priv = dev - > dev_private ;
2015-07-03 06:25:10 +08:00
WARN_ON ( ! mutex_is_locked ( & dev_priv - > fbc . lock ) ) ;
2014-12-09 00:09:10 +08:00
intel_fbc_cancel_work ( dev_priv ) ;
work = kzalloc ( sizeof ( * work ) , GFP_KERNEL ) ;
if ( work = = NULL ) {
DRM_ERROR ( " Failed to allocate FBC work structure \n " ) ;
2015-07-08 02:26:03 +08:00
dev_priv - > fbc . enable_fbc ( crtc ) ;
2014-12-09 00:09:10 +08:00
return ;
}
work - > crtc = crtc ;
work - > fb = crtc - > primary - > fb ;
INIT_DELAYED_WORK ( & work - > work , intel_fbc_work_fn ) ;
dev_priv - > fbc . fbc_work = work ;
/* Delay the actual enabling to let pageflipping cease and the
* display to settle before starting the compression . Note that
* this delay also serves a second purpose : it allows for a
* vblank to pass after disabling the FBC before we attempt
* to modify the control registers .
*
* A more complicated solution would involve tracking vblanks
* following the termination of the page - flipping sequence
* and indeed performing the enable as a co - routine and not
* waiting synchronously upon the vblank .
*
* WaFbcWaitForVBlankBeforeEnable : ilk , snb
*/
schedule_delayed_work ( & work - > work , msecs_to_jiffies ( 50 ) ) ;
}
2015-07-03 06:25:10 +08:00
static void __intel_fbc_disable ( struct drm_device * dev )
{
struct drm_i915_private * dev_priv = dev - > dev_private ;
WARN_ON ( ! mutex_is_locked ( & dev_priv - > fbc . lock ) ) ;
intel_fbc_cancel_work ( dev_priv ) ;
2015-07-08 02:26:03 +08:00
dev_priv - > fbc . disable_fbc ( dev ) ;
2015-07-03 06:25:10 +08:00
dev_priv - > fbc . crtc = NULL ;
}
2014-12-08 22:46:31 +08:00
/**
* intel_fbc_disable - disable FBC
* @ dev : the drm_device
*
* This function disables FBC .
*/
2014-12-09 00:09:10 +08:00
void intel_fbc_disable ( struct drm_device * dev )
{
struct drm_i915_private * dev_priv = dev - > dev_private ;
2015-07-08 02:26:03 +08:00
if ( ! dev_priv - > fbc . enable_fbc )
2015-07-04 02:40:54 +08:00
return ;
2015-07-03 06:25:10 +08:00
mutex_lock ( & dev_priv - > fbc . lock ) ;
__intel_fbc_disable ( dev ) ;
mutex_unlock ( & dev_priv - > fbc . lock ) ;
}
2014-12-09 00:09:10 +08:00
2015-07-03 06:25:10 +08:00
/*
* intel_fbc_disable_crtc - disable FBC if it ' s associated with crtc
* @ crtc : the CRTC
*
* This function disables FBC if it ' s associated with the provided CRTC .
*/
void intel_fbc_disable_crtc ( struct intel_crtc * crtc )
{
struct drm_device * dev = crtc - > base . dev ;
struct drm_i915_private * dev_priv = dev - > dev_private ;
2014-12-09 00:09:10 +08:00
2015-07-08 02:26:03 +08:00
if ( ! dev_priv - > fbc . enable_fbc )
2015-07-04 02:40:54 +08:00
return ;
2015-07-03 06:25:10 +08:00
mutex_lock ( & dev_priv - > fbc . lock ) ;
if ( dev_priv - > fbc . crtc = = crtc )
__intel_fbc_disable ( dev ) ;
mutex_unlock ( & dev_priv - > fbc . lock ) ;
2014-12-09 00:09:10 +08:00
}
2015-06-13 01:36:20 +08:00
const char * intel_no_fbc_reason_str ( enum no_fbc_reason reason )
{
switch ( reason ) {
case FBC_OK :
return " FBC enabled but currently disabled in hardware " ;
case FBC_UNSUPPORTED :
return " unsupported by this chipset " ;
case FBC_NO_OUTPUT :
return " no output " ;
case FBC_STOLEN_TOO_SMALL :
return " not enough stolen memory " ;
case FBC_UNSUPPORTED_MODE :
return " mode incompatible with compression " ;
case FBC_MODE_TOO_LARGE :
return " mode too large for compression " ;
case FBC_BAD_PLANE :
return " FBC unsupported on plane " ;
case FBC_NOT_TILED :
return " framebuffer not tiled or fenced " ;
case FBC_MULTIPLE_PIPES :
return " more than one pipe active " ;
case FBC_MODULE_PARAM :
return " disabled per module param " ;
case FBC_CHIP_DEFAULT :
return " disabled per chip default " ;
case FBC_ROTATION :
return " rotation unsupported " ;
default :
MISSING_CASE ( reason ) ;
return " unknown reason " ;
}
}
static void set_no_fbc_reason ( struct drm_i915_private * dev_priv ,
2014-12-09 00:09:10 +08:00
enum no_fbc_reason reason )
{
if ( dev_priv - > fbc . no_fbc_reason = = reason )
2015-06-13 01:36:20 +08:00
return ;
2014-12-09 00:09:10 +08:00
dev_priv - > fbc . no_fbc_reason = reason ;
2015-06-13 01:36:20 +08:00
DRM_DEBUG_KMS ( " Disabling FBC: %s \n " , intel_no_fbc_reason_str ( reason ) ) ;
2014-12-09 00:09:10 +08:00
}
2015-02-14 03:23:41 +08:00
static struct drm_crtc * intel_fbc_find_crtc ( struct drm_i915_private * dev_priv )
{
struct drm_crtc * crtc = NULL , * tmp_crtc ;
2015-02-14 03:23:42 +08:00
enum pipe pipe ;
2015-02-14 03:23:43 +08:00
bool pipe_a_only = false , one_pipe_only = false ;
2015-02-14 03:23:42 +08:00
if ( IS_HASWELL ( dev_priv ) | | INTEL_INFO ( dev_priv ) - > gen > = 8 )
pipe_a_only = true ;
2015-02-14 03:23:43 +08:00
else if ( INTEL_INFO ( dev_priv ) - > gen < = 4 )
one_pipe_only = true ;
2015-02-14 03:23:42 +08:00
for_each_pipe ( dev_priv , pipe ) {
tmp_crtc = dev_priv - > pipe_to_crtc_mapping [ pipe ] ;
2015-02-14 03:23:41 +08:00
if ( intel_crtc_active ( tmp_crtc ) & &
2015-04-21 22:12:53 +08:00
to_intel_plane_state ( tmp_crtc - > primary - > state ) - > visible ) {
2015-02-14 03:23:43 +08:00
if ( one_pipe_only & & crtc ) {
2015-06-13 01:36:20 +08:00
set_no_fbc_reason ( dev_priv , FBC_MULTIPLE_PIPES ) ;
2015-02-14 03:23:41 +08:00
return NULL ;
}
crtc = tmp_crtc ;
}
2015-02-14 03:23:42 +08:00
if ( pipe_a_only )
break ;
2015-02-14 03:23:41 +08:00
}
if ( ! crtc | | crtc - > primary - > fb = = NULL ) {
2015-06-13 01:36:20 +08:00
set_no_fbc_reason ( dev_priv , FBC_NO_OUTPUT ) ;
2015-02-14 03:23:41 +08:00
return NULL ;
}
return crtc ;
}
2015-07-03 06:25:08 +08:00
static int find_compression_threshold ( struct drm_device * dev ,
struct drm_mm_node * node ,
int size ,
int fb_cpp )
{
struct drm_i915_private * dev_priv = dev - > dev_private ;
int compression_threshold = 1 ;
int ret ;
/* HACK: This code depends on what we will do in *_enable_fbc. If that
* code changes , this code needs to change as well .
*
* The enable_fbc code will attempt to use one of our 2 compression
* thresholds , therefore , in that case , we only have 1 resort .
*/
/* Try to over-allocate to reduce reallocations and fragmentation. */
ret = i915_gem_stolen_insert_node ( dev_priv , node , size < < = 1 , 4096 ) ;
if ( ret = = 0 )
return compression_threshold ;
again :
/* HW's ability to limit the CFB is 1:4 */
if ( compression_threshold > 4 | |
( fb_cpp = = 2 & & compression_threshold = = 2 ) )
return 0 ;
ret = i915_gem_stolen_insert_node ( dev_priv , node , size > > = 1 , 4096 ) ;
if ( ret & & INTEL_INFO ( dev ) - > gen < = 4 ) {
return 0 ;
} else if ( ret ) {
compression_threshold < < = 1 ;
goto again ;
} else {
return compression_threshold ;
}
}
static int intel_fbc_alloc_cfb ( struct drm_device * dev , int size , int fb_cpp )
{
struct drm_i915_private * dev_priv = dev - > dev_private ;
struct drm_mm_node * uninitialized_var ( compressed_llb ) ;
int ret ;
ret = find_compression_threshold ( dev , & dev_priv - > fbc . compressed_fb ,
size , fb_cpp ) ;
if ( ! ret )
goto err_llb ;
else if ( ret > 1 ) {
DRM_INFO ( " Reducing the compressed framebuffer size. This may lead to less power savings than a non-reduced-size. Try to increase stolen memory size if available in BIOS. \n " ) ;
}
dev_priv - > fbc . threshold = ret ;
if ( INTEL_INFO ( dev_priv ) - > gen > = 5 )
I915_WRITE ( ILK_DPFC_CB_BASE , dev_priv - > fbc . compressed_fb . start ) ;
else if ( IS_GM45 ( dev ) ) {
I915_WRITE ( DPFC_CB_BASE , dev_priv - > fbc . compressed_fb . start ) ;
} else {
compressed_llb = kzalloc ( sizeof ( * compressed_llb ) , GFP_KERNEL ) ;
if ( ! compressed_llb )
goto err_fb ;
ret = i915_gem_stolen_insert_node ( dev_priv , compressed_llb ,
4096 , 4096 ) ;
if ( ret )
goto err_fb ;
dev_priv - > fbc . compressed_llb = compressed_llb ;
I915_WRITE ( FBC_CFB_BASE ,
dev_priv - > mm . stolen_base + dev_priv - > fbc . compressed_fb . start ) ;
I915_WRITE ( FBC_LL_BASE ,
dev_priv - > mm . stolen_base + compressed_llb - > start ) ;
}
dev_priv - > fbc . uncompressed_size = size ;
DRM_DEBUG_KMS ( " reserved %d bytes of contiguous stolen space for FBC \n " ,
size ) ;
return 0 ;
err_fb :
kfree ( compressed_llb ) ;
i915_gem_stolen_remove_node ( dev_priv , & dev_priv - > fbc . compressed_fb ) ;
err_llb :
pr_info_once ( " drm: not enough stolen space for compressed buffer (need %d more bytes), disabling. Hint: you may be able to increase stolen memory size in the BIOS to avoid this. \n " , size ) ;
return - ENOSPC ;
}
2015-07-03 06:25:10 +08:00
static void __intel_fbc_cleanup_cfb ( struct drm_device * dev )
2015-07-03 06:25:08 +08:00
{
struct drm_i915_private * dev_priv = dev - > dev_private ;
if ( dev_priv - > fbc . uncompressed_size = = 0 )
return ;
i915_gem_stolen_remove_node ( dev_priv , & dev_priv - > fbc . compressed_fb ) ;
if ( dev_priv - > fbc . compressed_llb ) {
i915_gem_stolen_remove_node ( dev_priv ,
dev_priv - > fbc . compressed_llb ) ;
kfree ( dev_priv - > fbc . compressed_llb ) ;
}
dev_priv - > fbc . uncompressed_size = 0 ;
}
2015-07-03 06:25:10 +08:00
void intel_fbc_cleanup_cfb ( struct drm_device * dev )
{
struct drm_i915_private * dev_priv = dev - > dev_private ;
2015-07-08 02:26:03 +08:00
if ( ! dev_priv - > fbc . enable_fbc )
2015-07-04 02:40:54 +08:00
return ;
2015-07-03 06:25:10 +08:00
mutex_lock ( & dev_priv - > fbc . lock ) ;
__intel_fbc_cleanup_cfb ( dev ) ;
mutex_unlock ( & dev_priv - > fbc . lock ) ;
}
2015-07-03 06:25:08 +08:00
static int intel_fbc_setup_cfb ( struct drm_device * dev , int size , int fb_cpp )
{
struct drm_i915_private * dev_priv = dev - > dev_private ;
if ( size < = dev_priv - > fbc . uncompressed_size )
return 0 ;
/* Release any current block */
2015-07-03 06:25:10 +08:00
__intel_fbc_cleanup_cfb ( dev ) ;
2015-07-03 06:25:08 +08:00
return intel_fbc_alloc_cfb ( dev , size , fb_cpp ) ;
}
2014-12-09 00:09:10 +08:00
/**
2015-07-03 06:25:10 +08:00
* __intel_fbc_update - enable / disable FBC as needed , unlocked
2014-12-09 00:09:10 +08:00
* @ dev : the drm_device
*
* Set up the framebuffer compression hardware at mode set time . We
* enable it if possible :
* - plane A only ( on pre - 965 )
* - no pixel mulitply / line duplication
* - no alpha buffer discard
* - no dual wide
* - framebuffer < = max_hdisplay in width , max_vdisplay in height
*
* We can ' t assume that any compression will take place ( worst case ) ,
* so the compressed buffer has to be the same size as the uncompressed
* one . It also must reside ( along with the line length buffer ) in
* stolen memory .
*
* We need to enable / disable FBC on a global basis .
*/
2015-07-03 06:25:10 +08:00
static void __intel_fbc_update ( struct drm_device * dev )
2014-12-09 00:09:10 +08:00
{
struct drm_i915_private * dev_priv = dev - > dev_private ;
2015-02-14 03:23:41 +08:00
struct drm_crtc * crtc = NULL ;
2014-12-09 00:09:10 +08:00
struct intel_crtc * intel_crtc ;
struct drm_framebuffer * fb ;
struct drm_i915_gem_object * obj ;
const struct drm_display_mode * adjusted_mode ;
unsigned int max_width , max_height ;
2015-07-03 06:25:10 +08:00
WARN_ON ( ! mutex_is_locked ( & dev_priv - > fbc . lock ) ) ;
2015-02-10 19:05:50 +08:00
/* disable framebuffer compression in vGPU */
if ( intel_vgpu_active ( dev ) )
i915 . enable_fbc = 0 ;
2015-02-10 00:46:27 +08:00
if ( i915 . enable_fbc < 0 ) {
2015-06-13 01:36:20 +08:00
set_no_fbc_reason ( dev_priv , FBC_CHIP_DEFAULT ) ;
2015-02-10 00:46:27 +08:00
goto out_disable ;
}
2015-03-25 03:40:09 +08:00
if ( ! i915 . enable_fbc ) {
2015-06-13 01:36:20 +08:00
set_no_fbc_reason ( dev_priv , FBC_MODULE_PARAM ) ;
2015-02-10 00:46:27 +08:00
goto out_disable ;
2014-12-09 00:09:10 +08:00
}
/*
* If FBC is already on , we just have to verify that we can
* keep it that way . . .
* Need to disable if :
* - more than one pipe is active
* - changing FBC params ( stride , fence , mode )
* - new fb is too large to fit in compressed buffer
* - going to an unsupported config ( interlace , pixel multiply , etc . )
*/
2015-02-14 03:23:41 +08:00
crtc = intel_fbc_find_crtc ( dev_priv ) ;
if ( ! crtc )
2014-12-09 00:09:10 +08:00
goto out_disable ;
intel_crtc = to_intel_crtc ( crtc ) ;
fb = crtc - > primary - > fb ;
obj = intel_fb_obj ( fb ) ;
2015-01-15 20:55:25 +08:00
adjusted_mode = & intel_crtc - > config - > base . adjusted_mode ;
2014-12-09 00:09:10 +08:00
if ( ( adjusted_mode - > flags & DRM_MODE_FLAG_INTERLACE ) | |
( adjusted_mode - > flags & DRM_MODE_FLAG_DBLSCAN ) ) {
2015-06-13 01:36:20 +08:00
set_no_fbc_reason ( dev_priv , FBC_UNSUPPORTED_MODE ) ;
2014-12-09 00:09:10 +08:00
goto out_disable ;
}
if ( INTEL_INFO ( dev ) - > gen > = 8 | | IS_HASWELL ( dev ) ) {
max_width = 4096 ;
max_height = 4096 ;
} else if ( IS_G4X ( dev ) | | INTEL_INFO ( dev ) - > gen > = 5 ) {
max_width = 4096 ;
max_height = 2048 ;
} else {
max_width = 2048 ;
max_height = 1536 ;
}
2015-01-15 20:55:25 +08:00
if ( intel_crtc - > config - > pipe_src_w > max_width | |
intel_crtc - > config - > pipe_src_h > max_height ) {
2015-06-13 01:36:20 +08:00
set_no_fbc_reason ( dev_priv , FBC_MODE_TOO_LARGE ) ;
2014-12-09 00:09:10 +08:00
goto out_disable ;
}
if ( ( INTEL_INFO ( dev ) - > gen < 4 | | HAS_DDI ( dev ) ) & &
intel_crtc - > plane ! = PLANE_A ) {
2015-06-13 01:36:20 +08:00
set_no_fbc_reason ( dev_priv , FBC_BAD_PLANE ) ;
2014-12-09 00:09:10 +08:00
goto out_disable ;
}
/* The use of a CPU fence is mandatory in order to detect writes
* by the CPU to the scanout and trigger updates to the FBC .
*/
if ( obj - > tiling_mode ! = I915_TILING_X | |
obj - > fence_reg = = I915_FENCE_REG_NONE ) {
2015-06-13 01:36:20 +08:00
set_no_fbc_reason ( dev_priv , FBC_NOT_TILED ) ;
2014-12-09 00:09:10 +08:00
goto out_disable ;
}
if ( INTEL_INFO ( dev ) - > gen < = 4 & & ! IS_G4X ( dev ) & &
2015-01-22 08:35:41 +08:00
crtc - > primary - > state - > rotation ! = BIT ( DRM_ROTATE_0 ) ) {
2015-06-13 01:36:20 +08:00
set_no_fbc_reason ( dev_priv , FBC_ROTATION ) ;
2014-12-09 00:09:10 +08:00
goto out_disable ;
}
/* If the kernel debugger is active, always disable compression */
if ( in_dbg_master ( ) )
goto out_disable ;
2015-07-03 06:25:08 +08:00
if ( intel_fbc_setup_cfb ( dev , obj - > base . size ,
drm_format_plane_cpp ( fb - > pixel_format , 0 ) ) ) {
2015-06-13 01:36:20 +08:00
set_no_fbc_reason ( dev_priv , FBC_STOLEN_TOO_SMALL ) ;
2014-12-09 00:09:10 +08:00
goto out_disable ;
}
/* If the scanout has not changed, don't modify the FBC settings.
* Note that we make the fundamental assumption that the fb - > obj
* cannot be unpinned ( and have its GTT offset and fence revoked )
* without first being decoupled from the scanout and FBC disabled .
*/
2015-02-10 00:46:29 +08:00
if ( dev_priv - > fbc . crtc = = intel_crtc & &
2014-12-09 00:09:10 +08:00
dev_priv - > fbc . fb_id = = fb - > base . id & &
dev_priv - > fbc . y = = crtc - > y )
return ;
if ( intel_fbc_enabled ( dev ) ) {
/* We update FBC along two paths, after changing fb/crtc
* configuration ( modeswitching ) and after page - flipping
* finishes . For the latter , we know that not only did
* we disable the FBC at the start of the page - flip
* sequence , but also more than one vblank has passed .
*
* For the former case of modeswitching , it is possible
* to switch between two FBC valid configurations
* instantaneously so we do need to disable the FBC
* before we can modify its control registers . We also
* have to wait for the next vblank for that to take
* effect . However , since we delay enabling FBC we can
* assume that a vblank has passed since disabling and
* that we can safely alter the registers in the deferred
* callback .
*
* In the scenario that we go from a valid to invalid
* and then back to valid FBC configuration we have
* no strict enforcement that a vblank occurred since
* disabling the FBC . However , along all current pipe
* disabling paths we do need to wait for a vblank at
* some point . And we wait before enabling FBC anyway .
*/
DRM_DEBUG_KMS ( " disabling active FBC for update \n " ) ;
2015-07-03 06:25:10 +08:00
__intel_fbc_disable ( dev ) ;
2014-12-09 00:09:10 +08:00
}
intel_fbc_enable ( crtc ) ;
dev_priv - > fbc . no_fbc_reason = FBC_OK ;
return ;
out_disable :
/* Multiple disables should be harmless */
if ( intel_fbc_enabled ( dev ) ) {
DRM_DEBUG_KMS ( " unsupported config, disabling FBC \n " ) ;
2015-07-03 06:25:10 +08:00
__intel_fbc_disable ( dev ) ;
2014-12-09 00:09:10 +08:00
}
2015-07-03 06:25:10 +08:00
__intel_fbc_cleanup_cfb ( dev ) ;
}
/*
* intel_fbc_update - enable / disable FBC as needed
* @ dev : the drm_device
*
* This function reevaluates the overall state and enables or disables FBC .
*/
void intel_fbc_update ( struct drm_device * dev )
{
struct drm_i915_private * dev_priv = dev - > dev_private ;
2015-07-08 02:26:03 +08:00
if ( ! dev_priv - > fbc . enable_fbc )
2015-07-04 02:40:54 +08:00
return ;
2015-07-03 06:25:10 +08:00
mutex_lock ( & dev_priv - > fbc . lock ) ;
__intel_fbc_update ( dev ) ;
mutex_unlock ( & dev_priv - > fbc . lock ) ;
2014-12-09 00:09:10 +08:00
}
2015-02-14 03:23:46 +08:00
void intel_fbc_invalidate ( struct drm_i915_private * dev_priv ,
unsigned int frontbuffer_bits ,
enum fb_op_origin origin )
{
struct drm_device * dev = dev_priv - > dev ;
unsigned int fbc_bits ;
2015-07-08 02:26:03 +08:00
if ( ! dev_priv - > fbc . enable_fbc )
2015-07-04 02:40:54 +08:00
return ;
2015-02-14 03:23:46 +08:00
if ( origin = = ORIGIN_GTT )
return ;
2015-07-03 06:25:10 +08:00
mutex_lock ( & dev_priv - > fbc . lock ) ;
2015-02-14 03:23:46 +08:00
if ( dev_priv - > fbc . enabled )
fbc_bits = INTEL_FRONTBUFFER_PRIMARY ( dev_priv - > fbc . crtc - > pipe ) ;
else if ( dev_priv - > fbc . fbc_work )
fbc_bits = INTEL_FRONTBUFFER_PRIMARY (
to_intel_crtc ( dev_priv - > fbc . fbc_work - > crtc ) - > pipe ) ;
else
fbc_bits = dev_priv - > fbc . possible_framebuffer_bits ;
dev_priv - > fbc . busy_bits | = ( fbc_bits & frontbuffer_bits ) ;
if ( dev_priv - > fbc . busy_bits )
2015-07-03 06:25:10 +08:00
__intel_fbc_disable ( dev ) ;
mutex_unlock ( & dev_priv - > fbc . lock ) ;
2015-02-14 03:23:46 +08:00
}
void intel_fbc_flush ( struct drm_i915_private * dev_priv ,
unsigned int frontbuffer_bits )
{
struct drm_device * dev = dev_priv - > dev ;
2015-07-08 02:26:03 +08:00
if ( ! dev_priv - > fbc . enable_fbc )
2015-07-04 02:40:54 +08:00
return ;
2015-07-03 06:25:10 +08:00
mutex_lock ( & dev_priv - > fbc . lock ) ;
2015-02-14 03:23:46 +08:00
if ( ! dev_priv - > fbc . busy_bits )
2015-07-03 06:25:10 +08:00
goto out ;
2015-02-14 03:23:46 +08:00
dev_priv - > fbc . busy_bits & = ~ frontbuffer_bits ;
if ( ! dev_priv - > fbc . busy_bits )
2015-07-03 06:25:10 +08:00
__intel_fbc_update ( dev ) ;
out :
mutex_unlock ( & dev_priv - > fbc . lock ) ;
2015-02-14 03:23:46 +08:00
}
2014-12-08 22:46:31 +08:00
/**
* intel_fbc_init - Initialize FBC
* @ dev_priv : the i915 device
*
* This function might be called during PM init process .
*/
2014-12-09 00:09:10 +08:00
void intel_fbc_init ( struct drm_i915_private * dev_priv )
{
2015-02-14 03:23:46 +08:00
enum pipe pipe ;
2015-07-03 06:25:10 +08:00
mutex_init ( & dev_priv - > fbc . lock ) ;
2014-12-09 00:09:10 +08:00
if ( ! HAS_FBC ( dev_priv ) ) {
dev_priv - > fbc . enabled = false ;
2015-02-10 00:46:28 +08:00
dev_priv - > fbc . no_fbc_reason = FBC_UNSUPPORTED ;
2014-12-09 00:09:10 +08:00
return ;
}
2015-02-14 03:23:46 +08:00
for_each_pipe ( dev_priv , pipe ) {
dev_priv - > fbc . possible_framebuffer_bits | =
INTEL_FRONTBUFFER_PRIMARY ( pipe ) ;
if ( IS_HASWELL ( dev_priv ) | | INTEL_INFO ( dev_priv ) - > gen > = 8 )
break ;
}
2014-12-09 00:09:10 +08:00
if ( INTEL_INFO ( dev_priv ) - > gen > = 7 ) {
2015-07-08 02:26:03 +08:00
dev_priv - > fbc . fbc_enabled = ilk_fbc_enabled ;
dev_priv - > fbc . enable_fbc = gen7_fbc_enable ;
dev_priv - > fbc . disable_fbc = ilk_fbc_disable ;
2014-12-09 00:09:10 +08:00
} else if ( INTEL_INFO ( dev_priv ) - > gen > = 5 ) {
2015-07-08 02:26:03 +08:00
dev_priv - > fbc . fbc_enabled = ilk_fbc_enabled ;
dev_priv - > fbc . enable_fbc = ilk_fbc_enable ;
dev_priv - > fbc . disable_fbc = ilk_fbc_disable ;
2014-12-09 00:09:10 +08:00
} else if ( IS_GM45 ( dev_priv ) ) {
2015-07-08 02:26:03 +08:00
dev_priv - > fbc . fbc_enabled = g4x_fbc_enabled ;
dev_priv - > fbc . enable_fbc = g4x_fbc_enable ;
dev_priv - > fbc . disable_fbc = g4x_fbc_disable ;
2014-12-09 00:09:10 +08:00
} else {
2015-07-08 02:26:03 +08:00
dev_priv - > fbc . fbc_enabled = i8xx_fbc_enabled ;
dev_priv - > fbc . enable_fbc = i8xx_fbc_enable ;
dev_priv - > fbc . disable_fbc = i8xx_fbc_disable ;
2014-12-09 00:09:10 +08:00
/* This value was pulled out of someone's hat */
I915_WRITE ( FBC_CONTROL , 500 < < FBC_CTL_INTERVAL_SHIFT ) ;
}
2015-07-08 02:26:03 +08:00
dev_priv - > fbc . enabled = dev_priv - > fbc . fbc_enabled ( dev_priv - > dev ) ;
2014-12-09 00:09:10 +08:00
}