Linux kernel mirror (for testing) git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel os linux

drm/amdgpu: fix vkms crtc settings

otherwise adev->mode_info.crtcs[] is NULL

Signed-off-by: Flora Cui <flora.cui@amd.com>
Reviewed-by: Alex Deucher <alexander.deucher@amd.com>
Signed-off-by: Alex Deucher <alexander.deucher@amd.com>

authored by

Flora Cui and committed by
Alex Deucher
deefd07e 4f7ee199

+30 -17
+28 -14
drivers/gpu/drm/amd/amdgpu/amdgpu_vkms.c
··· 16 16 #include "ivsrcid/ivsrcid_vislands30.h" 17 17 #include "amdgpu_vkms.h" 18 18 #include "amdgpu_display.h" 19 + #include "atom.h" 20 + #include "amdgpu_irq.h" 19 21 20 22 /** 21 23 * DOC: amdgpu_vkms ··· 43 41 44 42 static enum hrtimer_restart amdgpu_vkms_vblank_simulate(struct hrtimer *timer) 45 43 { 46 - struct amdgpu_vkms_output *output = container_of(timer, 47 - struct amdgpu_vkms_output, 48 - vblank_hrtimer); 49 - struct drm_crtc *crtc = &output->crtc; 44 + struct amdgpu_crtc *amdgpu_crtc = container_of(timer, struct amdgpu_crtc, vblank_timer); 45 + struct drm_crtc *crtc = &amdgpu_crtc->base; 46 + struct amdgpu_vkms_output *output = drm_crtc_to_amdgpu_vkms_output(crtc); 50 47 u64 ret_overrun; 51 48 bool ret; 52 49 53 - ret_overrun = hrtimer_forward_now(&output->vblank_hrtimer, 50 + ret_overrun = hrtimer_forward_now(&amdgpu_crtc->vblank_timer, 54 51 output->period_ns); 55 52 WARN_ON(ret_overrun != 1); 56 53 ··· 66 65 unsigned int pipe = drm_crtc_index(crtc); 67 66 struct drm_vblank_crtc *vblank = &dev->vblank[pipe]; 68 67 struct amdgpu_vkms_output *out = drm_crtc_to_amdgpu_vkms_output(crtc); 68 + struct amdgpu_crtc *amdgpu_crtc = to_amdgpu_crtc(crtc); 69 69 70 70 drm_calc_timestamping_constants(crtc, &crtc->mode); 71 71 72 - hrtimer_init(&out->vblank_hrtimer, CLOCK_MONOTONIC, HRTIMER_MODE_REL); 73 - out->vblank_hrtimer.function = &amdgpu_vkms_vblank_simulate; 74 72 out->period_ns = ktime_set(0, vblank->framedur_ns); 75 - hrtimer_start(&out->vblank_hrtimer, out->period_ns, HRTIMER_MODE_REL); 73 + hrtimer_start(&amdgpu_crtc->vblank_timer, out->period_ns, HRTIMER_MODE_REL); 76 74 77 75 return 0; 78 76 } 79 77 80 78 static void amdgpu_vkms_disable_vblank(struct drm_crtc *crtc) 81 79 { 82 - struct amdgpu_vkms_output *out = drm_crtc_to_amdgpu_vkms_output(crtc); 80 + struct amdgpu_crtc *amdgpu_crtc = to_amdgpu_crtc(crtc); 83 81 84 - hrtimer_cancel(&out->vblank_hrtimer); 82 + hrtimer_cancel(&amdgpu_crtc->vblank_timer); 85 83 } 86 84 87 85 static bool amdgpu_vkms_get_vblank_timestamp(struct drm_crtc *crtc, ··· 92 92 unsigned int pipe = crtc->index; 93 93 struct amdgpu_vkms_output *output = drm_crtc_to_amdgpu_vkms_output(crtc); 94 94 struct drm_vblank_crtc *vblank = &dev->vblank[pipe]; 95 + struct amdgpu_crtc *amdgpu_crtc = to_amdgpu_crtc(crtc); 95 96 96 97 if (!READ_ONCE(vblank->enabled)) { 97 98 *vblank_time = ktime_get(); 98 99 return true; 99 100 } 100 101 101 - *vblank_time = READ_ONCE(output->vblank_hrtimer.node.expires); 102 + *vblank_time = READ_ONCE(amdgpu_crtc->vblank_timer.node.expires); 102 103 103 104 if (WARN_ON(*vblank_time == vblank->time)) 104 105 return true; ··· 166 165 static int amdgpu_vkms_crtc_init(struct drm_device *dev, struct drm_crtc *crtc, 167 166 struct drm_plane *primary, struct drm_plane *cursor) 168 167 { 168 + struct amdgpu_device *adev = drm_to_adev(dev); 169 + struct amdgpu_crtc *amdgpu_crtc = to_amdgpu_crtc(crtc); 169 170 int ret; 170 171 171 172 ret = drm_crtc_init_with_planes(dev, crtc, primary, cursor, ··· 178 175 } 179 176 180 177 drm_crtc_helper_add(crtc, &amdgpu_vkms_crtc_helper_funcs); 178 + 179 + amdgpu_crtc->crtc_id = drm_crtc_index(crtc); 180 + adev->mode_info.crtcs[drm_crtc_index(crtc)] = amdgpu_crtc; 181 + 182 + amdgpu_crtc->pll_id = ATOM_PPLL_INVALID; 183 + amdgpu_crtc->encoder = NULL; 184 + amdgpu_crtc->connector = NULL; 185 + amdgpu_crtc->vsync_timer_enabled = AMDGPU_IRQ_STATE_DISABLE; 186 + 187 + hrtimer_init(&amdgpu_crtc->vblank_timer, CLOCK_MONOTONIC, HRTIMER_MODE_REL); 188 + amdgpu_crtc->vblank_timer.function = &amdgpu_vkms_vblank_simulate; 181 189 182 190 return ret; 183 191 } ··· 415 401 { 416 402 struct drm_connector *connector = &output->connector; 417 403 struct drm_encoder *encoder = &output->encoder; 418 - struct drm_crtc *crtc = &output->crtc; 404 + struct drm_crtc *crtc = &output->crtc.base; 419 405 struct drm_plane *primary, *cursor = NULL; 420 406 int ret; 421 407 ··· 518 504 int i = 0; 519 505 520 506 for (i = 0; i < adev->mode_info.num_crtc; i++) 521 - if (adev->amdgpu_vkms_output[i].vblank_hrtimer.function) 522 - hrtimer_cancel(&adev->amdgpu_vkms_output[i].vblank_hrtimer); 507 + if (adev->mode_info.crtcs[i]) 508 + hrtimer_cancel(&adev->mode_info.crtcs[i]->vblank_timer); 523 509 524 510 kfree(adev->mode_info.bios_hardcoded_edid); 525 511 kfree(adev->amdgpu_vkms_output);
+2 -3
drivers/gpu/drm/amd/amdgpu/amdgpu_vkms.h
··· 10 10 #define YRES_MAX 16384 11 11 12 12 #define drm_crtc_to_amdgpu_vkms_output(target) \ 13 - container_of(target, struct amdgpu_vkms_output, crtc) 13 + container_of(target, struct amdgpu_vkms_output, crtc.base) 14 14 15 15 extern const struct amdgpu_ip_block_version amdgpu_vkms_ip_block; 16 16 17 17 struct amdgpu_vkms_output { 18 - struct drm_crtc crtc; 18 + struct amdgpu_crtc crtc; 19 19 struct drm_encoder encoder; 20 20 struct drm_connector connector; 21 - struct hrtimer vblank_hrtimer; 22 21 ktime_t period_ns; 23 22 struct drm_pending_vblank_event *event; 24 23 };