mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	drm/vmwgfx: Add and connect CRTC helper functions
Atomic mode set requires us to refactor existing vmw_stdu_crtc_set_config code into sections that check the validity of the new mode, and sections that actually program the hardware state. vmw_du_crtc_atomic_check() takes CRTC-related checking code. In a later patch, vmw_du_primary_plane_atomic_check() will take framebuffer-related checking code. These helpers won't be called until we flip on the atomic support flag or set drm_crtc_funcs->set_config to using the atomic helper. v2: * The state->num_connector is actually the total number of potential connectors, not just the one associated with the display unit. The proper one to check is ->connector_mask. * Add the check to only allow plane state to be the same as crtc state (Thanks to mlankhorst) * Make sure to turn on SVGA mode before using VRAM. SVGA mode is disabled in master_drop if dbdev is not running. v3: * Moved dot clock override to crtc_atomic_check Signed-off-by: Sinclair Yeh <syeh@vmware.com> Signed-off-by: Thomas Hellstrom <thellstrom@vmware.com> Reviewed-by: Thomas Hellstrom <thellstrom@vmware.com> Acked-by: Daniel Vetter <daniel@ffwll.ch>
This commit is contained in:
		
							parent
							
								
									d7721ca711
								
							
						
					
					
						commit
						06ec41909e
					
				| @ -390,6 +390,61 @@ void vmw_du_primary_plane_destroy(struct drm_plane *plane) | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | int vmw_du_crtc_atomic_check(struct drm_crtc *crtc, | ||||||
|  | 			     struct drm_crtc_state *new_state) | ||||||
|  | { | ||||||
|  | 	struct vmw_display_unit *du = vmw_crtc_to_du(new_state->crtc); | ||||||
|  | 	int connector_mask = 1 << drm_connector_index(&du->connector); | ||||||
|  | 	bool has_primary = new_state->plane_mask & | ||||||
|  | 			   BIT(drm_plane_index(crtc->primary)); | ||||||
|  | 
 | ||||||
|  | 	/* We always want to have an active plane with an active CRTC */ | ||||||
|  | 	if (has_primary != new_state->enable) | ||||||
|  | 		return -EINVAL; | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 	if (new_state->connector_mask != connector_mask && | ||||||
|  | 	    new_state->connector_mask != 0) { | ||||||
|  | 		DRM_ERROR("Invalid connectors configuration\n"); | ||||||
|  | 		return -EINVAL; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/*
 | ||||||
|  | 	 * Our virtual device does not have a dot clock, so use the logical | ||||||
|  | 	 * clock value as the dot clock. | ||||||
|  | 	 */ | ||||||
|  | 	if (new_state->mode.crtc_clock == 0) | ||||||
|  | 		new_state->adjusted_mode.crtc_clock = new_state->mode.clock; | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | void vmw_du_crtc_atomic_begin(struct drm_crtc *crtc, | ||||||
|  | 			      struct drm_crtc_state *old_crtc_state) | ||||||
|  | { | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | void vmw_du_crtc_atomic_flush(struct drm_crtc *crtc, | ||||||
|  | 			      struct drm_crtc_state *old_crtc_state) | ||||||
|  | { | ||||||
|  | 	struct drm_pending_vblank_event *event = crtc->state->event; | ||||||
|  | 
 | ||||||
|  | 	if (event) { | ||||||
|  | 		crtc->state->event = NULL; | ||||||
|  | 
 | ||||||
|  | 		spin_lock_irq(&crtc->dev->event_lock); | ||||||
|  | 		if (drm_crtc_vblank_get(crtc) == 0) | ||||||
|  | 			drm_crtc_arm_vblank_event(crtc, event); | ||||||
|  | 		else | ||||||
|  | 			drm_crtc_send_vblank_event(crtc, event); | ||||||
|  | 		spin_unlock_irq(&crtc->dev->event_lock); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| /**
 | /**
 | ||||||
|  * vmw_du_crtc_duplicate_state - duplicate crtc state |  * vmw_du_crtc_duplicate_state - duplicate crtc state | ||||||
|  * @crtc: DRM crtc |  * @crtc: DRM crtc | ||||||
|  | |||||||
| @ -161,6 +161,7 @@ struct vmw_crtc_state { | |||||||
|  * @surf Display surface for STDU |  * @surf Display surface for STDU | ||||||
|  * @dmabuf display dmabuf for SOU |  * @dmabuf display dmabuf for SOU | ||||||
|  * @content_fb_type Used by STDU. |  * @content_fb_type Used by STDU. | ||||||
|  |  * @dmabuf_size Size of the dmabuf, used by Screen Object Display Unit | ||||||
|  * @pinned pin count for STDU display surface |  * @pinned pin count for STDU display surface | ||||||
|  */ |  */ | ||||||
| struct vmw_plane_state { | struct vmw_plane_state { | ||||||
| @ -169,6 +170,7 @@ struct vmw_plane_state { | |||||||
| 	struct vmw_dma_buffer *dmabuf; | 	struct vmw_dma_buffer *dmabuf; | ||||||
| 
 | 
 | ||||||
| 	int content_fb_type; | 	int content_fb_type; | ||||||
|  | 	unsigned long dmabuf_size; | ||||||
| 
 | 
 | ||||||
| 	int pinned; | 	int pinned; | ||||||
| }; | }; | ||||||
| @ -342,10 +344,17 @@ int vmw_du_cursor_plane_update(struct drm_plane *plane, | |||||||
| 			       uint32_t src_x, uint32_t src_y, | 			       uint32_t src_x, uint32_t src_y, | ||||||
| 			       uint32_t src_w, uint32_t src_h); | 			       uint32_t src_w, uint32_t src_h); | ||||||
| 
 | 
 | ||||||
|  | /* Atomic Helpers */ | ||||||
| void vmw_du_plane_reset(struct drm_plane *plane); | void vmw_du_plane_reset(struct drm_plane *plane); | ||||||
| struct drm_plane_state *vmw_du_plane_duplicate_state(struct drm_plane *plane); | struct drm_plane_state *vmw_du_plane_duplicate_state(struct drm_plane *plane); | ||||||
| void vmw_du_plane_destroy_state(struct drm_plane *plane, | void vmw_du_plane_destroy_state(struct drm_plane *plane, | ||||||
| 				struct drm_plane_state *state); | 				struct drm_plane_state *state); | ||||||
|  | int vmw_du_crtc_atomic_check(struct drm_crtc *crtc, | ||||||
|  | 			     struct drm_crtc_state *state); | ||||||
|  | void vmw_du_crtc_atomic_begin(struct drm_crtc *crtc, | ||||||
|  | 			      struct drm_crtc_state *old_crtc_state); | ||||||
|  | void vmw_du_crtc_atomic_flush(struct drm_crtc *crtc, | ||||||
|  | 			      struct drm_crtc_state *old_crtc_state); | ||||||
| void vmw_du_crtc_reset(struct drm_crtc *crtc); | void vmw_du_crtc_reset(struct drm_crtc *crtc); | ||||||
| struct drm_crtc_state *vmw_du_crtc_duplicate_state(struct drm_crtc *crtc); | struct drm_crtc_state *vmw_du_crtc_duplicate_state(struct drm_crtc *crtc); | ||||||
| void vmw_du_crtc_destroy_state(struct drm_crtc *crtc, | void vmw_du_crtc_destroy_state(struct drm_crtc *crtc, | ||||||
|  | |||||||
| @ -167,6 +167,7 @@ static int vmw_ldu_add_active(struct vmw_private *vmw_priv, | |||||||
| 	if (vfb != ld->fb) { | 	if (vfb != ld->fb) { | ||||||
| 		if (ld->fb && ld->fb->unpin) | 		if (ld->fb && ld->fb->unpin) | ||||||
| 			ld->fb->unpin(ld->fb); | 			ld->fb->unpin(ld->fb); | ||||||
|  | 		vmw_svga_enable(vmw_priv); | ||||||
| 		if (vfb->pin) | 		if (vfb->pin) | ||||||
| 			vfb->pin(vfb); | 			vfb->pin(vfb); | ||||||
| 		ld->fb = vfb; | 		ld->fb = vfb; | ||||||
| @ -190,6 +191,68 @@ static int vmw_ldu_add_active(struct vmw_private *vmw_priv, | |||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | /**
 | ||||||
|  |  * vmw_ldu_crtc_mode_set_nofb - Enable svga | ||||||
|  |  * | ||||||
|  |  * @crtc: CRTC associated with the new screen | ||||||
|  |  * | ||||||
|  |  * For LDU, just enable the svga | ||||||
|  |  */ | ||||||
|  | static void vmw_ldu_crtc_mode_set_nofb(struct drm_crtc *crtc) | ||||||
|  | { | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * vmw_ldu_crtc_helper_prepare - Noop | ||||||
|  |  * | ||||||
|  |  * @crtc: CRTC associated with the new screen | ||||||
|  |  * | ||||||
|  |  * Prepares the CRTC for a mode set, but we don't need to do anything here. | ||||||
|  |  * | ||||||
|  |  */ | ||||||
|  | static void vmw_ldu_crtc_helper_prepare(struct drm_crtc *crtc) | ||||||
|  | { | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * vmw_ldu_crtc_helper_commit - Noop | ||||||
|  |  * | ||||||
|  |  * @crtc: CRTC associated with the new screen | ||||||
|  |  * | ||||||
|  |  * This is called after a mode set has been completed.  Here's | ||||||
|  |  * usually a good place to call vmw_ldu_add_active/vmw_ldu_del_active | ||||||
|  |  * but since for LDU the display plane is closely tied to the | ||||||
|  |  * CRTC, it makes more sense to do those at plane update time. | ||||||
|  |  */ | ||||||
|  | static void vmw_ldu_crtc_helper_commit(struct drm_crtc *crtc) | ||||||
|  | { | ||||||
|  | 	struct vmw_private *dev_priv; | ||||||
|  | 	struct vmw_legacy_display_unit *ldu; | ||||||
|  | 	struct vmw_framebuffer *vfb; | ||||||
|  | 	struct drm_framebuffer *fb; | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 	ldu = vmw_crtc_to_ldu(crtc); | ||||||
|  | 	dev_priv = vmw_priv(crtc->dev); | ||||||
|  | 	fb       = crtc->primary->fb; | ||||||
|  | 
 | ||||||
|  | 	vfb = (fb) ? vmw_framebuffer_to_vfb(fb) : NULL; | ||||||
|  | 
 | ||||||
|  | 	if (vfb) | ||||||
|  | 		vmw_ldu_add_active(dev_priv, ldu, vfb); | ||||||
|  | 	else | ||||||
|  | 		vmw_ldu_del_active(dev_priv, ldu); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * vmw_ldu_crtc_helper_disable - Turns off CRTC | ||||||
|  |  * | ||||||
|  |  * @crtc: CRTC to be turned off | ||||||
|  |  */ | ||||||
|  | static void vmw_ldu_crtc_helper_disable(struct drm_crtc *crtc) | ||||||
|  | { | ||||||
|  | } | ||||||
|  | 
 | ||||||
| static int vmw_ldu_crtc_set_config(struct drm_mode_set *set) | static int vmw_ldu_crtc_set_config(struct drm_mode_set *set) | ||||||
| { | { | ||||||
| 	struct vmw_private *dev_priv; | 	struct vmw_private *dev_priv; | ||||||
| @ -346,6 +409,20 @@ static const struct drm_plane_funcs vmw_ldu_cursor_funcs = { | |||||||
| 	.atomic_destroy_state = vmw_du_plane_destroy_state, | 	.atomic_destroy_state = vmw_du_plane_destroy_state, | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | /*
 | ||||||
|  |  * Atomic Helpers | ||||||
|  |  */ | ||||||
|  | static const struct drm_crtc_helper_funcs vmw_ldu_crtc_helper_funcs = { | ||||||
|  | 	.prepare = vmw_ldu_crtc_helper_prepare, | ||||||
|  | 	.commit = vmw_ldu_crtc_helper_commit, | ||||||
|  | 	.disable = vmw_ldu_crtc_helper_disable, | ||||||
|  | 	.mode_set = drm_helper_crtc_mode_set, | ||||||
|  | 	.mode_set_nofb = vmw_ldu_crtc_mode_set_nofb, | ||||||
|  | 	.atomic_check = vmw_du_crtc_atomic_check, | ||||||
|  | 	.atomic_begin = vmw_du_crtc_atomic_begin, | ||||||
|  | 	.atomic_flush = vmw_du_crtc_atomic_flush, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
| 
 | 
 | ||||||
| static int vmw_ldu_init(struct vmw_private *dev_priv, unsigned unit) | static int vmw_ldu_init(struct vmw_private *dev_priv, unsigned unit) | ||||||
| { | { | ||||||
| @ -445,6 +522,8 @@ static int vmw_ldu_init(struct vmw_private *dev_priv, unsigned unit) | |||||||
| 		goto err_free_unregister; | 		goto err_free_unregister; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	drm_crtc_helper_add(crtc, &vmw_ldu_crtc_helper_funcs); | ||||||
|  | 
 | ||||||
| 	drm_mode_crtc_set_gamma_size(crtc, 256); | 	drm_mode_crtc_set_gamma_size(crtc, 256); | ||||||
| 
 | 
 | ||||||
| 	drm_object_attach_property(&connector->base, | 	drm_object_attach_property(&connector->base, | ||||||
|  | |||||||
| @ -250,6 +250,109 @@ static int vmw_sou_backing_alloc(struct vmw_private *dev_priv, | |||||||
| 	return ret; | 	return ret; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | /**
 | ||||||
|  |  * vmw_sou_crtc_mode_set_nofb - Create new screen | ||||||
|  |  * | ||||||
|  |  * @crtc: CRTC associated with the new screen | ||||||
|  |  * | ||||||
|  |  * This function creates/destroys a screen.  This function cannot fail, so if | ||||||
|  |  * somehow we run into a failure, just do the best we can to get out. | ||||||
|  |  */ | ||||||
|  | static void vmw_sou_crtc_mode_set_nofb(struct drm_crtc *crtc) | ||||||
|  | { | ||||||
|  | 	struct vmw_private *dev_priv; | ||||||
|  | 	struct vmw_screen_object_unit *sou; | ||||||
|  | 	struct vmw_framebuffer *vfb; | ||||||
|  | 	struct drm_framebuffer *fb; | ||||||
|  | 	struct drm_plane_state *ps; | ||||||
|  | 	struct vmw_plane_state *vps; | ||||||
|  | 	int ret; | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 	sou      = vmw_crtc_to_sou(crtc); | ||||||
|  | 	dev_priv = vmw_priv(crtc->dev); | ||||||
|  | 	ps       = crtc->primary->state; | ||||||
|  | 	fb       = ps->fb; | ||||||
|  | 	vps      = vmw_plane_state_to_vps(ps); | ||||||
|  | 
 | ||||||
|  | 	vfb = (fb) ? vmw_framebuffer_to_vfb(fb) : NULL; | ||||||
|  | 
 | ||||||
|  | 	if (sou->defined) { | ||||||
|  | 		ret = vmw_sou_fifo_destroy(dev_priv, sou); | ||||||
|  | 		if (ret) { | ||||||
|  | 			DRM_ERROR("Failed to destroy Screen Object\n"); | ||||||
|  | 			return; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (vfb) { | ||||||
|  | 		sou->buffer = vps->dmabuf; | ||||||
|  | 		sou->buffer_size = vps->dmabuf_size; | ||||||
|  | 
 | ||||||
|  | 		ret = vmw_sou_fifo_create(dev_priv, sou, crtc->x, crtc->y, | ||||||
|  | 					  &crtc->mode); | ||||||
|  | 		if (ret) | ||||||
|  | 			DRM_ERROR("Failed to define Screen Object %dx%d\n", | ||||||
|  | 				  crtc->x, crtc->y); | ||||||
|  | 
 | ||||||
|  | 		vmw_kms_add_active(dev_priv, &sou->base, vfb); | ||||||
|  | 	} else { | ||||||
|  | 		sou->buffer = NULL; | ||||||
|  | 		sou->buffer_size = 0; | ||||||
|  | 
 | ||||||
|  | 		vmw_kms_del_active(dev_priv, &sou->base); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * vmw_sou_crtc_helper_prepare - Noop | ||||||
|  |  * | ||||||
|  |  * @crtc: CRTC associated with the new screen | ||||||
|  |  * | ||||||
|  |  * Prepares the CRTC for a mode set, but we don't need to do anything here. | ||||||
|  |  */ | ||||||
|  | static void vmw_sou_crtc_helper_prepare(struct drm_crtc *crtc) | ||||||
|  | { | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * vmw_sou_crtc_helper_commit - Noop | ||||||
|  |  * | ||||||
|  |  * @crtc: CRTC associated with the new screen | ||||||
|  |  * | ||||||
|  |  * This is called after a mode set has been completed. | ||||||
|  |  */ | ||||||
|  | static void vmw_sou_crtc_helper_commit(struct drm_crtc *crtc) | ||||||
|  | { | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * vmw_sou_crtc_helper_disable - Turns off CRTC | ||||||
|  |  * | ||||||
|  |  * @crtc: CRTC to be turned off | ||||||
|  |  */ | ||||||
|  | static void vmw_sou_crtc_helper_disable(struct drm_crtc *crtc) | ||||||
|  | { | ||||||
|  | 	struct vmw_private *dev_priv; | ||||||
|  | 	struct vmw_screen_object_unit *sou; | ||||||
|  | 	int ret; | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 	if (!crtc) { | ||||||
|  | 		DRM_ERROR("CRTC is NULL\n"); | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	sou = vmw_crtc_to_sou(crtc); | ||||||
|  | 	dev_priv = vmw_priv(crtc->dev); | ||||||
|  | 
 | ||||||
|  | 	if (sou->defined) { | ||||||
|  | 		ret = vmw_sou_fifo_destroy(dev_priv, sou); | ||||||
|  | 		if (ret) | ||||||
|  | 			DRM_ERROR("Failed to destroy Screen Object\n"); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
| static int vmw_sou_crtc_set_config(struct drm_mode_set *set) | static int vmw_sou_crtc_set_config(struct drm_mode_set *set) | ||||||
| { | { | ||||||
| 	struct vmw_private *dev_priv; | 	struct vmw_private *dev_priv; | ||||||
| @ -527,6 +630,20 @@ static const struct drm_plane_funcs vmw_sou_cursor_funcs = { | |||||||
| 	.atomic_destroy_state = vmw_du_plane_destroy_state, | 	.atomic_destroy_state = vmw_du_plane_destroy_state, | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | /*
 | ||||||
|  |  * Atomic Helpers | ||||||
|  |  */ | ||||||
|  | static const struct drm_crtc_helper_funcs vmw_sou_crtc_helper_funcs = { | ||||||
|  | 	.prepare = vmw_sou_crtc_helper_prepare, | ||||||
|  | 	.commit = vmw_sou_crtc_helper_commit, | ||||||
|  | 	.disable = vmw_sou_crtc_helper_disable, | ||||||
|  | 	.mode_set = drm_helper_crtc_mode_set, | ||||||
|  | 	.mode_set_nofb = vmw_sou_crtc_mode_set_nofb, | ||||||
|  | 	.atomic_check = vmw_du_crtc_atomic_check, | ||||||
|  | 	.atomic_begin = vmw_du_crtc_atomic_begin, | ||||||
|  | 	.atomic_flush = vmw_du_crtc_atomic_flush, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
| 
 | 
 | ||||||
| static int vmw_sou_init(struct vmw_private *dev_priv, unsigned unit) | static int vmw_sou_init(struct vmw_private *dev_priv, unsigned unit) | ||||||
| { | { | ||||||
| @ -626,6 +743,8 @@ static int vmw_sou_init(struct vmw_private *dev_priv, unsigned unit) | |||||||
| 		goto err_free_unregister; | 		goto err_free_unregister; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	drm_crtc_helper_add(crtc, &vmw_sou_crtc_helper_funcs); | ||||||
|  | 
 | ||||||
| 	drm_mode_crtc_set_gamma_size(crtc, 256); | 	drm_mode_crtc_set_gamma_size(crtc, 256); | ||||||
| 
 | 
 | ||||||
| 	drm_object_attach_property(&connector->base, | 	drm_object_attach_property(&connector->base, | ||||||
|  | |||||||
| @ -500,6 +500,106 @@ out_srf_unref: | |||||||
| 	return ret; | 	return ret; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * vmw_stdu_crtc_mode_set_nofb - Updates screen target size | ||||||
|  |  * | ||||||
|  |  * @crtc: CRTC associated with the screen target | ||||||
|  |  * | ||||||
|  |  * This function defines/destroys a screen target | ||||||
|  |  * | ||||||
|  |  */ | ||||||
|  | static void vmw_stdu_crtc_mode_set_nofb(struct drm_crtc *crtc) | ||||||
|  | { | ||||||
|  | 	struct vmw_private *dev_priv; | ||||||
|  | 	struct vmw_screen_target_display_unit *stdu; | ||||||
|  | 	int ret; | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 	stdu     = vmw_crtc_to_stdu(crtc); | ||||||
|  | 	dev_priv = vmw_priv(crtc->dev); | ||||||
|  | 
 | ||||||
|  | 	if (stdu->defined) { | ||||||
|  | 		ret = vmw_stdu_bind_st(dev_priv, stdu, NULL); | ||||||
|  | 		if (ret) | ||||||
|  | 			DRM_ERROR("Failed to blank CRTC\n"); | ||||||
|  | 
 | ||||||
|  | 		(void) vmw_stdu_update_st(dev_priv, stdu); | ||||||
|  | 
 | ||||||
|  | 		ret = vmw_stdu_destroy_st(dev_priv, stdu); | ||||||
|  | 		if (ret) | ||||||
|  | 			DRM_ERROR("Failed to destroy Screen Target\n"); | ||||||
|  | 
 | ||||||
|  | 		stdu->content_fb_type = SAME_AS_DISPLAY; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (!crtc->state->enable) | ||||||
|  | 		return; | ||||||
|  | 
 | ||||||
|  | 	vmw_svga_enable(dev_priv); | ||||||
|  | 	ret = vmw_stdu_define_st(dev_priv, stdu, &crtc->mode, crtc->x, crtc->y); | ||||||
|  | 
 | ||||||
|  | 	if (ret) | ||||||
|  | 		DRM_ERROR("Failed to define Screen Target of size %dx%d\n", | ||||||
|  | 			  crtc->x, crtc->y); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | static void vmw_stdu_crtc_helper_prepare(struct drm_crtc *crtc) | ||||||
|  | { | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | static void vmw_stdu_crtc_helper_commit(struct drm_crtc *crtc) | ||||||
|  | { | ||||||
|  | 	struct vmw_private *dev_priv; | ||||||
|  | 	struct vmw_screen_target_display_unit *stdu; | ||||||
|  | 	struct vmw_framebuffer *vfb; | ||||||
|  | 	struct drm_framebuffer *fb; | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 	stdu     = vmw_crtc_to_stdu(crtc); | ||||||
|  | 	dev_priv = vmw_priv(crtc->dev); | ||||||
|  | 	fb       = crtc->primary->fb; | ||||||
|  | 
 | ||||||
|  | 	vfb = (fb) ? vmw_framebuffer_to_vfb(fb) : NULL; | ||||||
|  | 
 | ||||||
|  | 	if (vfb) | ||||||
|  | 		vmw_kms_add_active(dev_priv, &stdu->base, vfb); | ||||||
|  | 	else | ||||||
|  | 		vmw_kms_del_active(dev_priv, &stdu->base); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void vmw_stdu_crtc_helper_disable(struct drm_crtc *crtc) | ||||||
|  | { | ||||||
|  | 	struct vmw_private *dev_priv; | ||||||
|  | 	struct vmw_screen_target_display_unit *stdu; | ||||||
|  | 	int ret; | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 	if (!crtc) { | ||||||
|  | 		DRM_ERROR("CRTC is NULL\n"); | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	stdu     = vmw_crtc_to_stdu(crtc); | ||||||
|  | 	dev_priv = vmw_priv(crtc->dev); | ||||||
|  | 
 | ||||||
|  | 	if (stdu->defined) { | ||||||
|  | 		ret = vmw_stdu_bind_st(dev_priv, stdu, NULL); | ||||||
|  | 		if (ret) | ||||||
|  | 			DRM_ERROR("Failed to blank CRTC\n"); | ||||||
|  | 
 | ||||||
|  | 		(void) vmw_stdu_update_st(dev_priv, stdu); | ||||||
|  | 
 | ||||||
|  | 		ret = vmw_stdu_destroy_st(dev_priv, stdu); | ||||||
|  | 		if (ret) | ||||||
|  | 			DRM_ERROR("Failed to destroy Screen Target\n"); | ||||||
|  | 
 | ||||||
|  | 		stdu->content_fb_type = SAME_AS_DISPLAY; | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
| /**
 | /**
 | ||||||
|  * vmw_stdu_crtc_set_config - Sets a mode |  * vmw_stdu_crtc_set_config - Sets a mode | ||||||
|  * |  * | ||||||
| @ -1113,6 +1213,21 @@ static const struct drm_plane_funcs vmw_stdu_cursor_funcs = { | |||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | /*
 | ||||||
|  |  * Atomic Helpers | ||||||
|  |  */ | ||||||
|  | static const struct drm_crtc_helper_funcs vmw_stdu_crtc_helper_funcs = { | ||||||
|  | 	.prepare = vmw_stdu_crtc_helper_prepare, | ||||||
|  | 	.commit = vmw_stdu_crtc_helper_commit, | ||||||
|  | 	.disable = vmw_stdu_crtc_helper_disable, | ||||||
|  | 	.mode_set = drm_helper_crtc_mode_set, | ||||||
|  | 	.mode_set_nofb = vmw_stdu_crtc_mode_set_nofb, | ||||||
|  | 	.atomic_check = vmw_du_crtc_atomic_check, | ||||||
|  | 	.atomic_begin = vmw_du_crtc_atomic_begin, | ||||||
|  | 	.atomic_flush = vmw_du_crtc_atomic_flush, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| /**
 | /**
 | ||||||
|  * vmw_stdu_init - Sets up a Screen Target Display Unit |  * vmw_stdu_init - Sets up a Screen Target Display Unit | ||||||
|  * |  * | ||||||
| @ -1219,6 +1334,8 @@ static int vmw_stdu_init(struct vmw_private *dev_priv, unsigned unit) | |||||||
| 		goto err_free_unregister; | 		goto err_free_unregister; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	drm_crtc_helper_add(crtc, &vmw_stdu_crtc_helper_funcs); | ||||||
|  | 
 | ||||||
| 	drm_mode_crtc_set_gamma_size(crtc, 256); | 	drm_mode_crtc_set_gamma_size(crtc, 256); | ||||||
| 
 | 
 | ||||||
| 	drm_object_attach_property(&connector->base, | 	drm_object_attach_property(&connector->base, | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 Sinclair Yeh
						Sinclair Yeh