aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorAlex Deucher <alexander.deucher@amd.com>2013-04-10 09:47:05 -0400
committerAlex Deucher <alexander.deucher@amd.com>2013-04-22 10:39:07 -0400
commit2f86e2ede39a98650c2d465857405ef1c51372b1 (patch)
tree868a64891f2b902d4d902220c5066856be16388b
parent10257a6d8359c41407eb26b7ad7bf710a7e00155 (diff)
drm/radeon: properly lock disp in mc_stop/resume for r5xx-r7xx
Need to wait for the new addresses to take affect before re-enabling the MC. Signed-off-by: Alex Deucher <alexander.deucher@amd.com> Cc: stable@vger.kernel.org
-rw-r--r--drivers/gpu/drm/radeon/r500_reg.h1
-rw-r--r--drivers/gpu/drm/radeon/rv515.c43
2 files changed, 44 insertions, 0 deletions
diff --git a/drivers/gpu/drm/radeon/r500_reg.h b/drivers/gpu/drm/radeon/r500_reg.h
index c0dc8d3ba0bb..b52420902124 100644
--- a/drivers/gpu/drm/radeon/r500_reg.h
+++ b/drivers/gpu/drm/radeon/r500_reg.h
@@ -358,6 +358,7 @@
358#define AVIVO_D1CRTC_STATUS_HV_COUNT 0x60ac 358#define AVIVO_D1CRTC_STATUS_HV_COUNT 0x60ac
359#define AVIVO_D1CRTC_STEREO_CONTROL 0x60c4 359#define AVIVO_D1CRTC_STEREO_CONTROL 0x60c4
360 360
361#define AVIVO_D1MODE_MASTER_UPDATE_LOCK 0x60e0
361#define AVIVO_D1MODE_MASTER_UPDATE_MODE 0x60e4 362#define AVIVO_D1MODE_MASTER_UPDATE_MODE 0x60e4
362 363
363/* master controls */ 364/* master controls */
diff --git a/drivers/gpu/drm/radeon/rv515.c b/drivers/gpu/drm/radeon/rv515.c
index 5e1ba16c7a77..6a1e5dd5b5ee 100644
--- a/drivers/gpu/drm/radeon/rv515.c
+++ b/drivers/gpu/drm/radeon/rv515.c
@@ -338,6 +338,22 @@ void rv515_mc_stop(struct radeon_device *rdev, struct rv515_mc_save *save)
338 } 338 }
339 /* wait for the MC to settle */ 339 /* wait for the MC to settle */
340 udelay(100); 340 udelay(100);
341
342 /* lock double buffered regs */
343 for (i = 0; i < rdev->num_crtc; i++) {
344 if (save->crtc_enabled[i]) {
345 tmp = RREG32(AVIVO_D1GRPH_UPDATE + crtc_offsets[i]);
346 if (!(tmp & AVIVO_D1GRPH_UPDATE_LOCK)) {
347 tmp |= AVIVO_D1GRPH_UPDATE_LOCK;
348 WREG32(AVIVO_D1GRPH_UPDATE + crtc_offsets[i], tmp);
349 }
350 tmp = RREG32(AVIVO_D1MODE_MASTER_UPDATE_LOCK + crtc_offsets[i]);
351 if (!(tmp & 1)) {
352 tmp |= 1;
353 WREG32(AVIVO_D1MODE_MASTER_UPDATE_LOCK + crtc_offsets[i], tmp);
354 }
355 }
356 }
341} 357}
342 358
343void rv515_mc_resume(struct radeon_device *rdev, struct rv515_mc_save *save) 359void rv515_mc_resume(struct radeon_device *rdev, struct rv515_mc_save *save)
@@ -367,6 +383,33 @@ void rv515_mc_resume(struct radeon_device *rdev, struct rv515_mc_save *save)
367 } 383 }
368 WREG32(R_000310_VGA_MEMORY_BASE_ADDRESS, (u32)rdev->mc.vram_start); 384 WREG32(R_000310_VGA_MEMORY_BASE_ADDRESS, (u32)rdev->mc.vram_start);
369 385
386 /* unlock regs and wait for update */
387 for (i = 0; i < rdev->num_crtc; i++) {
388 if (save->crtc_enabled[i]) {
389 tmp = RREG32(AVIVO_D1MODE_MASTER_UPDATE_MODE + crtc_offsets[i]);
390 if ((tmp & 0x3) != 0) {
391 tmp &= ~0x3;
392 WREG32(AVIVO_D1MODE_MASTER_UPDATE_MODE + crtc_offsets[i], tmp);
393 }
394 tmp = RREG32(AVIVO_D1GRPH_UPDATE + crtc_offsets[i]);
395 if (tmp & AVIVO_D1GRPH_UPDATE_LOCK) {
396 tmp &= ~AVIVO_D1GRPH_UPDATE_LOCK;
397 WREG32(AVIVO_D1GRPH_UPDATE + crtc_offsets[i], tmp);
398 }
399 tmp = RREG32(AVIVO_D1MODE_MASTER_UPDATE_LOCK + crtc_offsets[i]);
400 if (tmp & 1) {
401 tmp &= ~1;
402 WREG32(AVIVO_D1MODE_MASTER_UPDATE_LOCK + crtc_offsets[i], tmp);
403 }
404 for (j = 0; j < rdev->usec_timeout; j++) {
405 tmp = RREG32(AVIVO_D1GRPH_UPDATE + crtc_offsets[i]);
406 if ((tmp & AVIVO_D1GRPH_SURFACE_UPDATE_PENDING) == 0)
407 break;
408 udelay(1);
409 }
410 }
411 }
412
370 if (rdev->family >= CHIP_R600) { 413 if (rdev->family >= CHIP_R600) {
371 /* unblackout the MC */ 414 /* unblackout the MC */
372 if (rdev->family >= CHIP_RV770) 415 if (rdev->family >= CHIP_RV770)