Commit 1b0da5a3 authored by David Zhang's avatar David Zhang Committed by Alex Deucher
Browse files

drm/amd/display: update cursor position to DMUB FW



[why]
To involve the cursor position into dirty rectangle calculation.

[how]
- separate plane and cursor update by different DMUB command
- send the cursor information while cursor updating, when updating
  cursor position/attribute, store cursor pos/attr to hubp, and
  notify dmub FW to exit psr before program cursor registers

Signed-off-by: default avatarDavid Zhang <dingchen.zhang@amd.com>
Acked-by: default avatarLeo Li <sunpeng.li@amd.com>
Reviewed-by: default avatarHarry Wentland <harry.wentland@amd.com>
Signed-off-by: default avatarAlex Deucher <alexander.deucher@amd.com>
parent 381b590c
Loading
Loading
Loading
Loading
+2 −0
Original line number Diff line number Diff line
@@ -1188,6 +1188,8 @@ void hubp1_cursor_set_position(
	uint32_t dst_x_offset;
	uint32_t cur_en = pos->enable ? 1 : 0;

	hubp->curs_pos = *pos;

	/*
	 * Guard aganst cursor_set_position() from being called with invalid
	 * attributes
+131 −0
Original line number Diff line number Diff line
@@ -51,6 +51,8 @@
#include "link_hwss.h"
#include "dpcd_defs.h"
#include "dsc.h"
#include "dce/dmub_psr.h"
#include "dc_dmub_srv.h"
#include "dce/dmub_hw_lock_mgr.h"
#include "dc_trace.h"
#include "dce/dmub_outbox.h"
@@ -3328,6 +3330,115 @@ static bool dcn10_can_pipe_disable_cursor(struct pipe_ctx *pipe_ctx)
	return false;
}

static void dcn10_dmub_update_cursor_data(
		struct pipe_ctx *pipe_ctx,
		struct hubp *hubp,
		const struct dc_cursor_mi_param *param,
		const struct dc_cursor_position *cur_pos,
		const struct dc_cursor_attributes *cur_attr)
{
	union dmub_rb_cmd cmd;
	struct dmub_cmd_update_cursor_info_data *update_cursor_info;
	const struct dc_cursor_position *pos;
	const struct dc_cursor_attributes *attr;
	int src_x_offset = 0;
	int src_y_offset = 0;
	int x_hotspot = 0;
	int cursor_height = 0;
	int cursor_width = 0;
	uint32_t cur_en = 0;
	unsigned int panel_inst = 0;

	struct dc_debug_options *debug = &hubp->ctx->dc->debug;

	if (!debug->enable_sw_cntl_psr && pipe_ctx->stream->link->psr_settings.psr_version != DC_PSR_VERSION_SU_1)
		return;

	if (pipe_ctx->stream->link->psr_settings.psr_version == DC_PSR_VERSION_UNSUPPORTED ||
		pipe_ctx->plane_state->address.type == PLN_ADDR_TYPE_VIDEO_PROGRESSIVE)
		return;

	/**
	 * if cur_pos == NULL means the caller is from cursor_set_attribute
	 * then driver use previous cursor position data
	 * if cur_attr == NULL means the caller is from cursor_set_position
	 * then driver use previous cursor attribute
	 * if cur_pos or cur_attr is not NULL then update it
	 */
	if (cur_pos != NULL)
		pos = cur_pos;
	else
		pos = &hubp->curs_pos;

	if (cur_attr != NULL)
		attr = cur_attr;
	else
		attr = &hubp->curs_attr;

	if (!dc_get_edp_link_panel_inst(hubp->ctx->dc, pipe_ctx->stream->link, &panel_inst))
		return;

	src_x_offset = pos->x - pos->x_hotspot - param->viewport.x;
	src_y_offset = pos->y - pos->y_hotspot - param->viewport.y;
	x_hotspot = pos->x_hotspot;
	cursor_height = (int)attr->height;
	cursor_width = (int)attr->width;
	cur_en = pos->enable ? 1:0;

	// Rotated cursor width/height and hotspots tweaks for offset calculation
	if (param->rotation == ROTATION_ANGLE_90 || param->rotation == ROTATION_ANGLE_270) {
		swap(cursor_height, cursor_width);
		if (param->rotation == ROTATION_ANGLE_90) {
			src_x_offset = pos->x - pos->y_hotspot - param->viewport.x;
			src_y_offset = pos->y - pos->x_hotspot - param->viewport.y;
		}
	} else if (param->rotation == ROTATION_ANGLE_180) {
		src_x_offset = pos->x - param->viewport.x;
		src_y_offset = pos->y - param->viewport.y;
	}

	if (param->mirror) {
		x_hotspot = param->viewport.width - x_hotspot;
		src_x_offset = param->viewport.x + param->viewport.width - src_x_offset;
	}

	if (src_x_offset >= (int)param->viewport.width)
		cur_en = 0;  /* not visible beyond right edge*/

	if (src_x_offset + cursor_width <= 0)
		cur_en = 0;  /* not visible beyond left edge*/

	if (src_y_offset >= (int)param->viewport.height)
		cur_en = 0;  /* not visible beyond bottom edge*/

	if (src_y_offset + cursor_height <= 0)
		cur_en = 0;  /* not visible beyond top edge*/

	// Cursor bitmaps have different hotspot values
	// There's a possibility that the above logic returns a negative value, so we clamp them to 0
	if (src_x_offset < 0)
		src_x_offset = 0;
	if (src_y_offset < 0)
		src_y_offset = 0;

	memset(&cmd, 0x0, sizeof(cmd));
	cmd.update_cursor_info.header.type = DMUB_CMD__UPDATE_CURSOR_INFO;
	cmd.update_cursor_info.header.payload_bytes =
			sizeof(cmd.update_cursor_info.update_cursor_info_data);
	update_cursor_info = &cmd.update_cursor_info.update_cursor_info_data;
	update_cursor_info->cursor_rect.x = src_x_offset + param->viewport.x;
	update_cursor_info->cursor_rect.y = src_y_offset + param->viewport.y;
	update_cursor_info->cursor_rect.width = attr->width;
	update_cursor_info->cursor_rect.height = attr->height;
	update_cursor_info->enable = cur_en;
	update_cursor_info->pipe_idx = pipe_ctx->pipe_idx;
	update_cursor_info->cmd_version = DMUB_CMD_PSR_CONTROL_VERSION_1;
	update_cursor_info->panel_inst = panel_inst;
	dc_dmub_srv_cmd_queue(pipe_ctx->stream->ctx->dmub_srv, &cmd);
	dc_dmub_srv_cmd_execute(pipe_ctx->stream->ctx->dmub_srv);
	dc_dmub_srv_wait_idle(pipe_ctx->stream->ctx->dmub_srv);
}

void dcn10_set_cursor_position(struct pipe_ctx *pipe_ctx)
{
	struct dc_cursor_position pos_cpy = pipe_ctx->stream->cursor_position;
@@ -3526,6 +3637,7 @@ void dcn10_set_cursor_position(struct pipe_ctx *pipe_ctx)
			pipe_ctx->plane_res.scl_data.viewport.height - pos_cpy.y;
	}

	dcn10_dmub_update_cursor_data(pipe_ctx, hubp, &param, &pos_cpy, NULL);
	hubp->funcs->set_cursor_position(hubp, &pos_cpy, &param);
	dpp->funcs->set_cursor_position(dpp, &pos_cpy, &param, hubp->curs_attr.width, hubp->curs_attr.height);
}
@@ -3533,6 +3645,25 @@ void dcn10_set_cursor_position(struct pipe_ctx *pipe_ctx)
void dcn10_set_cursor_attribute(struct pipe_ctx *pipe_ctx)
{
	struct dc_cursor_attributes *attributes = &pipe_ctx->stream->cursor_attributes;
	struct dc_cursor_mi_param param = { 0 };

	/**
	 * If enter PSR without cursor attribute update
	 * the cursor attribute of dmub_restore_plane
	 * are initial value. call dmub to exit PSR and
	 * restore plane then update cursor attribute to
	 * avoid override with initial value
	 */
	if (pipe_ctx->plane_state != NULL) {
		param.pixel_clk_khz = pipe_ctx->stream->timing.pix_clk_100hz / 10;
		param.ref_clk_khz = pipe_ctx->stream->ctx->dc->res_pool->ref_clocks.dchub_ref_clock_inKhz;
		param.viewport = pipe_ctx->plane_res.scl_data.viewport;
		param.h_scale_ratio = pipe_ctx->plane_res.scl_data.ratios.horz;
		param.v_scale_ratio = pipe_ctx->plane_res.scl_data.ratios.vert;
		param.rotation = pipe_ctx->plane_state->rotation;
		param.mirror = pipe_ctx->plane_state->horizontal_mirror;
		dcn10_dmub_update_cursor_data(pipe_ctx, pipe_ctx->plane_res.hubp, &param, NULL, attributes);
	}

	pipe_ctx->plane_res.hubp->funcs->set_cursor_attributes(
			pipe_ctx->plane_res.hubp, attributes);
+2 −0
Original line number Diff line number Diff line
@@ -967,6 +967,8 @@ void hubp2_cursor_set_position(
	uint32_t dst_x_offset;
	uint32_t cur_en = pos->enable ? 1 : 0;

	hubp->curs_pos = *pos;

	/*
	 * Guard aganst cursor_set_position() from being called with invalid
	 * attributes
+1 −0
Original line number Diff line number Diff line
@@ -63,6 +63,7 @@ struct hubp {
	int opp_id;
	int mpcc_id;
	struct dc_cursor_attributes curs_attr;
	struct dc_cursor_position curs_pos;
	bool power_gated;
};