|
@@ -2663,231 +2663,6 @@ void r600_dma_fini(struct radeon_device *rdev)
|
|
radeon_ring_fini(rdev, &rdev->ring[R600_RING_TYPE_DMA_INDEX]);
|
|
radeon_ring_fini(rdev, &rdev->ring[R600_RING_TYPE_DMA_INDEX]);
|
|
}
|
|
}
|
|
|
|
|
|
-/*
|
|
|
|
- * UVD
|
|
|
|
- */
|
|
|
|
-uint32_t r600_uvd_get_rptr(struct radeon_device *rdev,
|
|
|
|
- struct radeon_ring *ring)
|
|
|
|
-{
|
|
|
|
- return RREG32(UVD_RBC_RB_RPTR);
|
|
|
|
-}
|
|
|
|
-
|
|
|
|
-uint32_t r600_uvd_get_wptr(struct radeon_device *rdev,
|
|
|
|
- struct radeon_ring *ring)
|
|
|
|
-{
|
|
|
|
- return RREG32(UVD_RBC_RB_WPTR);
|
|
|
|
-}
|
|
|
|
-
|
|
|
|
-void r600_uvd_set_wptr(struct radeon_device *rdev,
|
|
|
|
- struct radeon_ring *ring)
|
|
|
|
-{
|
|
|
|
- WREG32(UVD_RBC_RB_WPTR, ring->wptr);
|
|
|
|
-}
|
|
|
|
-
|
|
|
|
-static int r600_uvd_rbc_start(struct radeon_device *rdev, bool ring_test)
|
|
|
|
-{
|
|
|
|
- struct radeon_ring *ring = &rdev->ring[R600_RING_TYPE_UVD_INDEX];
|
|
|
|
- uint32_t rb_bufsz, tmp;
|
|
|
|
- int r;
|
|
|
|
-
|
|
|
|
- /* force RBC into idle state */
|
|
|
|
- WREG32(UVD_RBC_RB_CNTL, 0x11010101);
|
|
|
|
-
|
|
|
|
- /* Set the write pointer delay */
|
|
|
|
- WREG32(UVD_RBC_RB_WPTR_CNTL, 0);
|
|
|
|
-
|
|
|
|
- /* programm the 4GB memory segment for rptr and ring buffer */
|
|
|
|
- WREG32(UVD_LMI_EXT40_ADDR, upper_32_bits(ring->gpu_addr) |
|
|
|
|
- (0x7 << 16) | (0x1 << 31));
|
|
|
|
-
|
|
|
|
- /* Initialize the ring buffer's read and write pointers */
|
|
|
|
- WREG32(UVD_RBC_RB_RPTR, 0x0);
|
|
|
|
-
|
|
|
|
- ring->wptr = ring->rptr = RREG32(UVD_RBC_RB_RPTR);
|
|
|
|
- WREG32(UVD_RBC_RB_WPTR, ring->wptr);
|
|
|
|
-
|
|
|
|
- /* set the ring address */
|
|
|
|
- WREG32(UVD_RBC_RB_BASE, ring->gpu_addr);
|
|
|
|
-
|
|
|
|
- /* Set ring buffer size */
|
|
|
|
- rb_bufsz = drm_order(ring->ring_size);
|
|
|
|
- rb_bufsz = (0x1 << 8) | rb_bufsz;
|
|
|
|
- WREG32_P(UVD_RBC_RB_CNTL, rb_bufsz, ~0x11f1f);
|
|
|
|
-
|
|
|
|
- if (ring_test) {
|
|
|
|
- ring->ready = true;
|
|
|
|
- r = radeon_ring_test(rdev, R600_RING_TYPE_UVD_INDEX, ring);
|
|
|
|
- if (r) {
|
|
|
|
- ring->ready = false;
|
|
|
|
- return r;
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- r = radeon_ring_lock(rdev, ring, 10);
|
|
|
|
- if (r) {
|
|
|
|
- DRM_ERROR("radeon: ring failed to lock UVD ring (%d).\n", r);
|
|
|
|
- return r;
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- tmp = PACKET0(UVD_SEMA_WAIT_FAULT_TIMEOUT_CNTL, 0);
|
|
|
|
- radeon_ring_write(ring, tmp);
|
|
|
|
- radeon_ring_write(ring, 0xFFFFF);
|
|
|
|
-
|
|
|
|
- tmp = PACKET0(UVD_SEMA_WAIT_INCOMPLETE_TIMEOUT_CNTL, 0);
|
|
|
|
- radeon_ring_write(ring, tmp);
|
|
|
|
- radeon_ring_write(ring, 0xFFFFF);
|
|
|
|
-
|
|
|
|
- tmp = PACKET0(UVD_SEMA_SIGNAL_INCOMPLETE_TIMEOUT_CNTL, 0);
|
|
|
|
- radeon_ring_write(ring, tmp);
|
|
|
|
- radeon_ring_write(ring, 0xFFFFF);
|
|
|
|
-
|
|
|
|
- /* Clear timeout status bits */
|
|
|
|
- radeon_ring_write(ring, PACKET0(UVD_SEMA_TIMEOUT_STATUS, 0));
|
|
|
|
- radeon_ring_write(ring, 0x8);
|
|
|
|
-
|
|
|
|
- radeon_ring_write(ring, PACKET0(UVD_SEMA_CNTL, 0));
|
|
|
|
- radeon_ring_write(ring, 3);
|
|
|
|
-
|
|
|
|
- radeon_ring_unlock_commit(rdev, ring);
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- return 0;
|
|
|
|
-}
|
|
|
|
-
|
|
|
|
-void r600_do_uvd_stop(struct radeon_device *rdev)
|
|
|
|
-{
|
|
|
|
- /* force RBC into idle state */
|
|
|
|
- WREG32(UVD_RBC_RB_CNTL, 0x11010101);
|
|
|
|
-
|
|
|
|
- /* Stall UMC and register bus before resetting VCPU */
|
|
|
|
- WREG32_P(UVD_LMI_CTRL2, 1 << 8, ~(1 << 8));
|
|
|
|
- WREG32_P(UVD_RB_ARB_CTRL, 1 << 3, ~(1 << 3));
|
|
|
|
- mdelay(1);
|
|
|
|
-
|
|
|
|
- /* put VCPU into reset */
|
|
|
|
- WREG32(UVD_SOFT_RESET, VCPU_SOFT_RESET);
|
|
|
|
- mdelay(5);
|
|
|
|
-
|
|
|
|
- /* disable VCPU clock */
|
|
|
|
- WREG32(UVD_VCPU_CNTL, 0x0);
|
|
|
|
-
|
|
|
|
- /* Unstall UMC and register bus */
|
|
|
|
- WREG32_P(UVD_LMI_CTRL2, 0, ~(1 << 8));
|
|
|
|
- WREG32_P(UVD_RB_ARB_CTRL, 0, ~(1 << 3));
|
|
|
|
-}
|
|
|
|
-
|
|
|
|
-void r600_uvd_stop(struct radeon_device *rdev)
|
|
|
|
-{
|
|
|
|
- struct radeon_ring *ring = &rdev->ring[R600_RING_TYPE_UVD_INDEX];
|
|
|
|
-
|
|
|
|
- r600_do_uvd_stop(rdev);
|
|
|
|
- ring->ready = false;
|
|
|
|
-}
|
|
|
|
-
|
|
|
|
-int r600_uvd_init(struct radeon_device *rdev, bool ring_test)
|
|
|
|
-{
|
|
|
|
- int i, j, r;
|
|
|
|
- /* disable byte swapping */
|
|
|
|
- u32 lmi_swap_cntl = 0;
|
|
|
|
- u32 mp_swap_cntl = 0;
|
|
|
|
-
|
|
|
|
- /* raise clocks while booting up the VCPU */
|
|
|
|
- radeon_set_uvd_clocks(rdev, 53300, 40000);
|
|
|
|
-
|
|
|
|
- /* disable clock gating */
|
|
|
|
- WREG32(UVD_CGC_GATE, 0);
|
|
|
|
-
|
|
|
|
- /* disable interupt */
|
|
|
|
- WREG32_P(UVD_MASTINT_EN, 0, ~(1 << 1));
|
|
|
|
-
|
|
|
|
- /* Stall UMC and register bus before resetting VCPU */
|
|
|
|
- WREG32_P(UVD_LMI_CTRL2, 1 << 8, ~(1 << 8));
|
|
|
|
- WREG32_P(UVD_RB_ARB_CTRL, 1 << 3, ~(1 << 3));
|
|
|
|
- mdelay(1);
|
|
|
|
-
|
|
|
|
- /* put LMI, VCPU, RBC etc... into reset */
|
|
|
|
- WREG32(UVD_SOFT_RESET, LMI_SOFT_RESET | VCPU_SOFT_RESET |
|
|
|
|
- LBSI_SOFT_RESET | RBC_SOFT_RESET | CSM_SOFT_RESET |
|
|
|
|
- CXW_SOFT_RESET | TAP_SOFT_RESET | LMI_UMC_SOFT_RESET);
|
|
|
|
- mdelay(5);
|
|
|
|
-
|
|
|
|
- /* take UVD block out of reset */
|
|
|
|
- WREG32_P(SRBM_SOFT_RESET, 0, ~SOFT_RESET_UVD);
|
|
|
|
- mdelay(5);
|
|
|
|
-
|
|
|
|
- /* initialize UVD memory controller */
|
|
|
|
- WREG32(UVD_LMI_CTRL, 0x40 | (1 << 8) | (1 << 13) |
|
|
|
|
- (1 << 21) | (1 << 9) | (1 << 20));
|
|
|
|
-
|
|
|
|
-#ifdef __BIG_ENDIAN
|
|
|
|
- /* swap (8 in 32) RB and IB */
|
|
|
|
- lmi_swap_cntl = 0xa;
|
|
|
|
- mp_swap_cntl = 0;
|
|
|
|
-#endif
|
|
|
|
- WREG32(UVD_LMI_SWAP_CNTL, lmi_swap_cntl);
|
|
|
|
- WREG32(UVD_MP_SWAP_CNTL, mp_swap_cntl);
|
|
|
|
-
|
|
|
|
- WREG32(UVD_MPC_SET_MUXA0, 0x40c2040);
|
|
|
|
- WREG32(UVD_MPC_SET_MUXA1, 0x0);
|
|
|
|
- WREG32(UVD_MPC_SET_MUXB0, 0x40c2040);
|
|
|
|
- WREG32(UVD_MPC_SET_MUXB1, 0x0);
|
|
|
|
- WREG32(UVD_MPC_SET_ALU, 0);
|
|
|
|
- WREG32(UVD_MPC_SET_MUX, 0x88);
|
|
|
|
-
|
|
|
|
- /* take all subblocks out of reset, except VCPU */
|
|
|
|
- WREG32(UVD_SOFT_RESET, VCPU_SOFT_RESET);
|
|
|
|
- mdelay(5);
|
|
|
|
-
|
|
|
|
- /* enable VCPU clock */
|
|
|
|
- WREG32(UVD_VCPU_CNTL, 1 << 9);
|
|
|
|
-
|
|
|
|
- /* enable UMC */
|
|
|
|
- WREG32_P(UVD_LMI_CTRL2, 0, ~(1 << 8));
|
|
|
|
-
|
|
|
|
- /* boot up the VCPU */
|
|
|
|
- WREG32(UVD_SOFT_RESET, 0);
|
|
|
|
- mdelay(10);
|
|
|
|
-
|
|
|
|
- WREG32_P(UVD_RB_ARB_CTRL, 0, ~(1 << 3));
|
|
|
|
-
|
|
|
|
- for (i = 0; i < 10; ++i) {
|
|
|
|
- uint32_t status;
|
|
|
|
- for (j = 0; j < 100; ++j) {
|
|
|
|
- status = RREG32(UVD_STATUS);
|
|
|
|
- if (status & 2)
|
|
|
|
- break;
|
|
|
|
- mdelay(10);
|
|
|
|
- }
|
|
|
|
- r = 0;
|
|
|
|
- if (status & 2)
|
|
|
|
- break;
|
|
|
|
-
|
|
|
|
- DRM_ERROR("UVD not responding, trying to reset the VCPU!!!\n");
|
|
|
|
- WREG32_P(UVD_SOFT_RESET, VCPU_SOFT_RESET, ~VCPU_SOFT_RESET);
|
|
|
|
- mdelay(10);
|
|
|
|
- WREG32_P(UVD_SOFT_RESET, 0, ~VCPU_SOFT_RESET);
|
|
|
|
- mdelay(10);
|
|
|
|
- r = -1;
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- if (r) {
|
|
|
|
- DRM_ERROR("UVD not responding, giving up!!!\n");
|
|
|
|
- goto done;
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- /* enable interupt */
|
|
|
|
- WREG32_P(UVD_MASTINT_EN, 3<<1, ~(3 << 1));
|
|
|
|
-
|
|
|
|
- r = r600_uvd_rbc_start(rdev, ring_test);
|
|
|
|
- if (!r)
|
|
|
|
- DRM_INFO("UVD initialized successfully.\n");
|
|
|
|
-
|
|
|
|
-done:
|
|
|
|
- /* lower clocks again */
|
|
|
|
- radeon_set_uvd_clocks(rdev, 0, 0);
|
|
|
|
-
|
|
|
|
- return r;
|
|
|
|
-}
|
|
|
|
-
|
|
|
|
/*
|
|
/*
|
|
* GPU scratch registers helpers function.
|
|
* GPU scratch registers helpers function.
|
|
*/
|
|
*/
|
|
@@ -2997,40 +2772,6 @@ int r600_dma_ring_test(struct radeon_device *rdev,
|
|
return r;
|
|
return r;
|
|
}
|
|
}
|
|
|
|
|
|
-int r600_uvd_ring_test(struct radeon_device *rdev, struct radeon_ring *ring)
|
|
|
|
-{
|
|
|
|
- uint32_t tmp = 0;
|
|
|
|
- unsigned i;
|
|
|
|
- int r;
|
|
|
|
-
|
|
|
|
- WREG32(UVD_CONTEXT_ID, 0xCAFEDEAD);
|
|
|
|
- r = radeon_ring_lock(rdev, ring, 3);
|
|
|
|
- if (r) {
|
|
|
|
- DRM_ERROR("radeon: cp failed to lock ring %d (%d).\n",
|
|
|
|
- ring->idx, r);
|
|
|
|
- return r;
|
|
|
|
- }
|
|
|
|
- radeon_ring_write(ring, PACKET0(UVD_CONTEXT_ID, 0));
|
|
|
|
- radeon_ring_write(ring, 0xDEADBEEF);
|
|
|
|
- radeon_ring_unlock_commit(rdev, ring);
|
|
|
|
- for (i = 0; i < rdev->usec_timeout; i++) {
|
|
|
|
- tmp = RREG32(UVD_CONTEXT_ID);
|
|
|
|
- if (tmp == 0xDEADBEEF)
|
|
|
|
- break;
|
|
|
|
- DRM_UDELAY(1);
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- if (i < rdev->usec_timeout) {
|
|
|
|
- DRM_INFO("ring test on %d succeeded in %d usecs\n",
|
|
|
|
- ring->idx, i);
|
|
|
|
- } else {
|
|
|
|
- DRM_ERROR("radeon: ring %d test failed (0x%08X)\n",
|
|
|
|
- ring->idx, tmp);
|
|
|
|
- r = -EINVAL;
|
|
|
|
- }
|
|
|
|
- return r;
|
|
|
|
-}
|
|
|
|
-
|
|
|
|
/*
|
|
/*
|
|
* CP fences/semaphores
|
|
* CP fences/semaphores
|
|
*/
|
|
*/
|
|
@@ -3082,30 +2823,6 @@ void r600_fence_ring_emit(struct radeon_device *rdev,
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
-void r600_uvd_fence_emit(struct radeon_device *rdev,
|
|
|
|
- struct radeon_fence *fence)
|
|
|
|
-{
|
|
|
|
- struct radeon_ring *ring = &rdev->ring[fence->ring];
|
|
|
|
- uint64_t addr = rdev->fence_drv[fence->ring].gpu_addr;
|
|
|
|
-
|
|
|
|
- radeon_ring_write(ring, PACKET0(UVD_CONTEXT_ID, 0));
|
|
|
|
- radeon_ring_write(ring, fence->seq);
|
|
|
|
- radeon_ring_write(ring, PACKET0(UVD_GPCOM_VCPU_DATA0, 0));
|
|
|
|
- radeon_ring_write(ring, addr & 0xffffffff);
|
|
|
|
- radeon_ring_write(ring, PACKET0(UVD_GPCOM_VCPU_DATA1, 0));
|
|
|
|
- radeon_ring_write(ring, upper_32_bits(addr) & 0xff);
|
|
|
|
- radeon_ring_write(ring, PACKET0(UVD_GPCOM_VCPU_CMD, 0));
|
|
|
|
- radeon_ring_write(ring, 0);
|
|
|
|
-
|
|
|
|
- radeon_ring_write(ring, PACKET0(UVD_GPCOM_VCPU_DATA0, 0));
|
|
|
|
- radeon_ring_write(ring, 0);
|
|
|
|
- radeon_ring_write(ring, PACKET0(UVD_GPCOM_VCPU_DATA1, 0));
|
|
|
|
- radeon_ring_write(ring, 0);
|
|
|
|
- radeon_ring_write(ring, PACKET0(UVD_GPCOM_VCPU_CMD, 0));
|
|
|
|
- radeon_ring_write(ring, 2);
|
|
|
|
- return;
|
|
|
|
-}
|
|
|
|
-
|
|
|
|
void r600_semaphore_ring_emit(struct radeon_device *rdev,
|
|
void r600_semaphore_ring_emit(struct radeon_device *rdev,
|
|
struct radeon_ring *ring,
|
|
struct radeon_ring *ring,
|
|
struct radeon_semaphore *semaphore,
|
|
struct radeon_semaphore *semaphore,
|
|
@@ -3175,23 +2892,6 @@ void r600_dma_semaphore_ring_emit(struct radeon_device *rdev,
|
|
radeon_ring_write(ring, upper_32_bits(addr) & 0xff);
|
|
radeon_ring_write(ring, upper_32_bits(addr) & 0xff);
|
|
}
|
|
}
|
|
|
|
|
|
-void r600_uvd_semaphore_emit(struct radeon_device *rdev,
|
|
|
|
- struct radeon_ring *ring,
|
|
|
|
- struct radeon_semaphore *semaphore,
|
|
|
|
- bool emit_wait)
|
|
|
|
-{
|
|
|
|
- uint64_t addr = semaphore->gpu_addr;
|
|
|
|
-
|
|
|
|
- radeon_ring_write(ring, PACKET0(UVD_SEMA_ADDR_LOW, 0));
|
|
|
|
- radeon_ring_write(ring, (addr >> 3) & 0x000FFFFF);
|
|
|
|
-
|
|
|
|
- radeon_ring_write(ring, PACKET0(UVD_SEMA_ADDR_HIGH, 0));
|
|
|
|
- radeon_ring_write(ring, (addr >> 23) & 0x000FFFFF);
|
|
|
|
-
|
|
|
|
- radeon_ring_write(ring, PACKET0(UVD_SEMA_CMD, 0));
|
|
|
|
- radeon_ring_write(ring, emit_wait ? 1 : 0);
|
|
|
|
-}
|
|
|
|
-
|
|
|
|
/**
|
|
/**
|
|
* r600_copy_cpdma - copy pages using the CP DMA engine
|
|
* r600_copy_cpdma - copy pages using the CP DMA engine
|
|
*
|
|
*
|
|
@@ -3656,16 +3356,6 @@ void r600_ring_ib_execute(struct radeon_device *rdev, struct radeon_ib *ib)
|
|
radeon_ring_write(ring, ib->length_dw);
|
|
radeon_ring_write(ring, ib->length_dw);
|
|
}
|
|
}
|
|
|
|
|
|
-void r600_uvd_ib_execute(struct radeon_device *rdev, struct radeon_ib *ib)
|
|
|
|
-{
|
|
|
|
- struct radeon_ring *ring = &rdev->ring[ib->ring];
|
|
|
|
-
|
|
|
|
- radeon_ring_write(ring, PACKET0(UVD_RBC_IB_BASE, 0));
|
|
|
|
- radeon_ring_write(ring, ib->gpu_addr);
|
|
|
|
- radeon_ring_write(ring, PACKET0(UVD_RBC_IB_SIZE, 0));
|
|
|
|
- radeon_ring_write(ring, ib->length_dw);
|
|
|
|
-}
|
|
|
|
-
|
|
|
|
int r600_ib_test(struct radeon_device *rdev, struct radeon_ring *ring)
|
|
int r600_ib_test(struct radeon_device *rdev, struct radeon_ring *ring)
|
|
{
|
|
{
|
|
struct radeon_ib ib;
|
|
struct radeon_ib ib;
|
|
@@ -3783,41 +3473,6 @@ int r600_dma_ib_test(struct radeon_device *rdev, struct radeon_ring *ring)
|
|
return r;
|
|
return r;
|
|
}
|
|
}
|
|
|
|
|
|
-int r600_uvd_ib_test(struct radeon_device *rdev, struct radeon_ring *ring)
|
|
|
|
-{
|
|
|
|
- struct radeon_fence *fence = NULL;
|
|
|
|
- int r;
|
|
|
|
-
|
|
|
|
- r = radeon_set_uvd_clocks(rdev, 53300, 40000);
|
|
|
|
- if (r) {
|
|
|
|
- DRM_ERROR("radeon: failed to raise UVD clocks (%d).\n", r);
|
|
|
|
- return r;
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- r = radeon_uvd_get_create_msg(rdev, ring->idx, 1, NULL);
|
|
|
|
- if (r) {
|
|
|
|
- DRM_ERROR("radeon: failed to get create msg (%d).\n", r);
|
|
|
|
- goto error;
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- r = radeon_uvd_get_destroy_msg(rdev, ring->idx, 1, &fence);
|
|
|
|
- if (r) {
|
|
|
|
- DRM_ERROR("radeon: failed to get destroy ib (%d).\n", r);
|
|
|
|
- goto error;
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- r = radeon_fence_wait(fence, false);
|
|
|
|
- if (r) {
|
|
|
|
- DRM_ERROR("radeon: fence wait failed (%d).\n", r);
|
|
|
|
- goto error;
|
|
|
|
- }
|
|
|
|
- DRM_INFO("ib test on ring %d succeeded\n", ring->idx);
|
|
|
|
-error:
|
|
|
|
- radeon_fence_unref(&fence);
|
|
|
|
- radeon_set_uvd_clocks(rdev, 0, 0);
|
|
|
|
- return r;
|
|
|
|
-}
|
|
|
|
-
|
|
|
|
/**
|
|
/**
|
|
* r600_dma_ring_ib_execute - Schedule an IB on the DMA engine
|
|
* r600_dma_ring_ib_execute - Schedule an IB on the DMA engine
|
|
*
|
|
*
|