diff options
-rw-r--r-- | src/radeon_dri2.c | 89 |
1 files changed, 75 insertions, 14 deletions
diff --git a/src/radeon_dri2.c b/src/radeon_dri2.c index 4cafbc65..ec52f6a8 100644 --- a/src/radeon_dri2.c +++ b/src/radeon_dri2.c @@ -71,7 +71,7 @@ radeon_dri2_create_buffers(DrawablePtr drawable, struct dri2_buffer_priv *privates; PixmapPtr pixmap, depth_pixmap; struct radeon_exa_pixmap_priv *driver_priv; - int i, r; + int i, r, need_enlarge = 0; int flags = 0; buffers = calloc(count, sizeof *buffers); @@ -100,7 +100,6 @@ radeon_dri2_create_buffers(DrawablePtr drawable, /* tile the back buffer */ switch(attachments[i]) { case DRI2BufferDepth: - case DRI2BufferDepthStencil: if (info->ChipFamily >= CHIP_FAMILY_R600) /* macro is the preferred setting, but the 2D detiling for software * fallbacks in mesa still has issues on some configurations @@ -109,6 +108,17 @@ radeon_dri2_create_buffers(DrawablePtr drawable, else flags = RADEON_CREATE_PIXMAP_TILING_MACRO | RADEON_CREATE_PIXMAP_TILING_MICRO; break; + case DRI2BufferDepthStencil: + if (info->ChipFamily >= CHIP_FAMILY_R600) { + /* macro is the preferred setting, but the 2D detiling for software + * fallbacks in mesa still has issues on some configurations + */ + flags = RADEON_CREATE_PIXMAP_TILING_MICRO; + if (info->ChipFamily >= CHIP_FAMILY_CEDAR) + need_enlarge = 1; + } else + flags = RADEON_CREATE_PIXMAP_TILING_MACRO | RADEON_CREATE_PIXMAP_TILING_MICRO; + break; case DRI2BufferBackLeft: case DRI2BufferBackRight: case DRI2BufferFakeFrontLeft: @@ -124,11 +134,31 @@ radeon_dri2_create_buffers(DrawablePtr drawable, default: flags = 0; } - pixmap = (*pScreen->CreatePixmap)(pScreen, - drawable->width, - drawable->height, - drawable->depth, - flags); + + if (need_enlarge) { + /* evergreen uses separate allocations for depth and stencil + * so we make an extra large depth buffer to cover stencil + * as well. + */ + int pitch = drawable->width * (drawable->depth / 8); + int aligned_height = (drawable->height + 7) & ~7; + int size = pitch * aligned_height; + size = (size + 255) & ~255; + size += drawable->width * aligned_height; + aligned_height = ((size / pitch) + 7) & ~7; + + pixmap = (*pScreen->CreatePixmap)(pScreen, + drawable->width, + aligned_height, + drawable->depth, + flags); + + } else + pixmap = (*pScreen->CreatePixmap)(pScreen, + drawable->width, + drawable->height, + drawable->depth, + flags); } if (attachments[i] == DRI2BufferDepth) { @@ -165,7 +195,7 @@ radeon_dri2_create_buffer(DrawablePtr drawable, struct dri2_buffer_priv *privates; PixmapPtr pixmap, depth_pixmap; struct radeon_exa_pixmap_priv *driver_priv; - int r; + int r, need_enlarge = 0; int flags; buffers = calloc(1, sizeof *buffers); @@ -194,7 +224,6 @@ radeon_dri2_create_buffer(DrawablePtr drawable, /* tile the back buffer */ switch(attachment) { case DRI2BufferDepth: - case DRI2BufferDepthStencil: /* macro is the preferred setting, but the 2D detiling for software * fallbacks in mesa still has issues on some configurations */ @@ -203,6 +232,17 @@ radeon_dri2_create_buffer(DrawablePtr drawable, else flags = RADEON_CREATE_PIXMAP_TILING_MACRO | RADEON_CREATE_PIXMAP_TILING_MICRO; break; + case DRI2BufferDepthStencil: + /* macro is the preferred setting, but the 2D detiling for software + * fallbacks in mesa still has issues on some configurations + */ + if (info->ChipFamily >= CHIP_FAMILY_R600) { + flags = RADEON_CREATE_PIXMAP_TILING_MICRO; + if (info->ChipFamily >= CHIP_FAMILY_CEDAR) + need_enlarge = 1; + } else + flags = RADEON_CREATE_PIXMAP_TILING_MACRO | RADEON_CREATE_PIXMAP_TILING_MICRO; + break; case DRI2BufferBackLeft: case DRI2BufferBackRight: case DRI2BufferFakeFrontLeft: @@ -218,11 +258,32 @@ radeon_dri2_create_buffer(DrawablePtr drawable, default: flags = 0; } - pixmap = (*pScreen->CreatePixmap)(pScreen, - drawable->width, - drawable->height, - (format != 0)?format:drawable->depth, - flags); + + if (need_enlarge) { + /* evergreen uses separate allocations for depth and stencil + * so we make an extra large depth buffer to cover stencil + * as well. + */ + int depth = (format != 0) ? format : drawable->depth; + int pitch = drawable->width * (depth / 8); + int aligned_height = (drawable->height + 7) & ~7; + int size = pitch * aligned_height; + size = (size + 255) & ~255; + size += drawable->width * aligned_height; + aligned_height = ((size / pitch) + 7) & ~7; + + pixmap = (*pScreen->CreatePixmap)(pScreen, + drawable->width, + aligned_height, + (format != 0)?format:drawable->depth, + flags); + + } else + pixmap = (*pScreen->CreatePixmap)(pScreen, + drawable->width, + drawable->height, + (format != 0)?format:drawable->depth, + flags); } if (attachment == DRI2BufferDepth) { |