Commit f0558542 authored by Dmytro Laktyushkin's avatar Dmytro Laktyushkin Committed by Alex Deucher
Browse files
parent 4c6e7598
Loading
Loading
Loading
Loading
+1 −0
Original line number Diff line number Diff line
@@ -517,6 +517,7 @@ static void split_stream_across_pipes(
	secondary_pipe->stream = primary_pipe->stream;
	secondary_pipe->tg = primary_pipe->tg;

	secondary_pipe->mpcc = pool->mpcc[secondary_pipe->pipe_idx];
	secondary_pipe->mi = pool->mis[secondary_pipe->pipe_idx];
	secondary_pipe->ipp = pool->ipps[secondary_pipe->pipe_idx];
	secondary_pipe->xfm = pool->transforms[secondary_pipe->pipe_idx];
+4 −6
Original line number Diff line number Diff line
@@ -1008,8 +1008,6 @@ static int acquire_first_split_pipe(

		if (pipe_ctx->top_pipe &&
				pipe_ctx->top_pipe->surface == pipe_ctx->surface) {
			int mpc_idx = pipe_ctx->mpc_idx;

			pipe_ctx->top_pipe->bottom_pipe = pipe_ctx->bottom_pipe;
			if (pipe_ctx->bottom_pipe)
				pipe_ctx->bottom_pipe->top_pipe = pipe_ctx->top_pipe;
@@ -1021,8 +1019,8 @@ static int acquire_first_split_pipe(
			pipe_ctx->xfm = pool->transforms[i];
			pipe_ctx->opp = pool->opps[i];
			pipe_ctx->dis_clk = pool->display_clock;
			pipe_ctx->mpcc = pool->mpcc[i];
			pipe_ctx->pipe_idx = i;
			pipe_ctx->mpc_idx = mpc_idx;

			pipe_ctx->stream = stream;
			return i;
@@ -1243,6 +1241,9 @@ static int acquire_first_free_pipe(
		if (!res_ctx->pipe_ctx[i].stream) {
			struct pipe_ctx *pipe_ctx = &res_ctx->pipe_ctx[i];

#if defined(CONFIG_DRM_AMD_DC_DCN1_0)
			pipe_ctx->mpcc = pool->mpcc[i];
#endif
			pipe_ctx->tg = pool->timing_generators[i];
			pipe_ctx->mi = pool->mis[i];
			pipe_ctx->ipp = pool->ipps[i];
@@ -1251,9 +1252,6 @@ static int acquire_first_free_pipe(
			pipe_ctx->dis_clk = pool->display_clock;
			pipe_ctx->pipe_idx = i;

#if defined(CONFIG_DRM_AMD_DC_DCN1_0)
			pipe_ctx->mpc_idx = -1;
#endif

			pipe_ctx->stream = stream;
			return i;
+7 −3
Original line number Diff line number Diff line
@@ -135,8 +135,11 @@ uint32_t generic_reg_wait(const struct dc_context *ctx,
	uint32_t reg_val;
	int i;

	if (ctx->dce_environment == DCE_ENV_FPGA_MAXIMUS)
		time_out_num_tries *= 20;
	if (IS_FPGA_MAXIMUS_DC(ctx->dce_environment)) {
		/* 35 seconds */
		delay_between_poll_us = 35000;
		time_out_num_tries = 1000;
	}

	for (i = 0; i <= time_out_num_tries; i++) {
		if (i) {
@@ -157,7 +160,8 @@ uint32_t generic_reg_wait(const struct dc_context *ctx,

	dm_error("REG_WAIT timeout %dus * %d tries - %s\n",
			delay_between_poll_us, time_out_num_tries, func_name);
	if (ctx->dce_environment != DCE_ENV_FPGA_MAXIMUS)

	if (!IS_FPGA_MAXIMUS_DC(ctx->dce_environment))
		BREAK_TO_DEBUGGER();

	return reg_val;
+38 −67
Original line number Diff line number Diff line
@@ -592,9 +592,19 @@ static void init_hw(struct core_dc *dc)
	for (i = 0; i < dc->res_pool->pipe_count; i++) {
		struct timing_generator *tg =
				dc->res_pool->timing_generators[i];
		struct mpcc *mpcc =
				dc->res_pool->mpcc[i];
		struct mpcc_cfg mpcc_cfg;

		lock_otg_master_update(dc->ctx, tg->inst);
		mpcc_cfg.opp_id = 0xf;
		mpcc_cfg.top_dpp_id = 0xf;
		mpcc_cfg.bot_mpcc_id = 0xf;
		mpcc_cfg.top_of_tree = true;
		mpcc->funcs->set(mpcc, &mpcc_cfg);
		unlock_otg_master(dc->ctx, tg->inst);

		tg->funcs->disable_vga(tg);

		/* Blank controller using driver code instead of
		 * command table.
		 */
@@ -819,8 +829,7 @@ static void reset_front_end_for_pipe(
		struct pipe_ctx *pipe_ctx,
		struct validate_context *context)
{
	struct dcn10_mpc *mpc = TO_DCN10_MPC(dc->res_pool->mpc);
	struct mpc_tree_cfg *tree_cfg = NULL;
	struct mpcc_cfg mpcc_cfg;

	if (!pipe_ctx->surface)
		return;
@@ -829,20 +838,14 @@ static void reset_front_end_for_pipe(

	lock_otg_master_update(dc->ctx, pipe_ctx->tg->inst);

	/* TODO: build stream pipes group id. For now, use stream otg
	 * id as pipe group id
	 */
	tree_cfg = &dc->current_context->res_ctx.mpc_tree[pipe_ctx->mpc_idx];

	if (!dcn10_remove_dpp(mpc, tree_cfg, pipe_ctx->pipe_idx)) {
		dm_logger_write(dc->ctx->logger, LOG_RESOURCE,
			"%s: failed to find dpp to be removed!\n",
			__func__);
	}
	mpcc_cfg.opp_id = 0xf;
	mpcc_cfg.top_dpp_id = 0xf;
	mpcc_cfg.bot_mpcc_id = 0xf;
	mpcc_cfg.top_of_tree = !pipe_ctx->top_pipe;
	pipe_ctx->mpcc->funcs->set(pipe_ctx->mpcc, &mpcc_cfg);

	pipe_ctx->top_pipe = NULL;
	pipe_ctx->bottom_pipe = NULL;
	pipe_ctx->mpc_idx = -1;

	unlock_master_tg_and_wait(dc->ctx, pipe_ctx->tg->inst);

@@ -850,8 +853,6 @@ static void reset_front_end_for_pipe(

	wait_no_outstanding_request(dc->ctx, pipe_ctx->pipe_idx);

	wait_mpcc_idle(mpc, pipe_ctx->pipe_idx);

	disable_clocks(dc->ctx, pipe_ctx->pipe_idx);

	pipe_ctx->xfm->funcs->transform_reset(pipe_ctx->xfm);
@@ -893,14 +894,10 @@ static void reset_hw_ctx_wrap(
	reset_hw_ctx(dc, context, reset_front_end_for_pipe);
	/* Reset Back End*/
	reset_hw_ctx(dc, context, reset_back_end_for_pipe);

	memcpy(context->res_ctx.mpc_tree,
			dc->current_context->res_ctx.mpc_tree,
			sizeof(struct mpc_tree_cfg) * dc->res_pool->pipe_count);
}

static bool patch_address_for_sbs_tb_stereo(struct pipe_ctx *pipe_ctx,
											PHYSICAL_ADDRESS_LOC *addr)
static bool patch_address_for_sbs_tb_stereo(
		struct pipe_ctx *pipe_ctx, PHYSICAL_ADDRESS_LOC *addr)
{
	struct core_surface *surface = pipe_ctx->surface;
	bool sec_split = pipe_ctx->top_pipe &&
@@ -1670,14 +1667,10 @@ static void update_dchubp_dpp(
	struct input_pixel_processor *ipp = pipe_ctx->ipp;
	struct core_surface *surface = pipe_ctx->surface;
	union plane_size size = surface->public.plane_size;
	struct mpc_tree_cfg *tree_cfg = NULL;
	struct default_adjustment ocsc = {0};
	enum dc_color_space color_space;
	struct tg_color black_color = {0};
	struct dcn10_mpc *mpc = TO_DCN10_MPC(dc->res_pool->mpc);
	struct pipe_ctx *temp_pipe;
	int i;
	int tree_pos = 0;
	struct mpcc_cfg mpcc_cfg;
	struct pipe_ctx *top_pipe;
	bool per_pixel_alpha = surface->public.per_pixel_alpha && pipe_ctx->bottom_pipe;

	/* TODO: proper fix once fpga works */
@@ -1716,39 +1709,23 @@ static void update_dchubp_dpp(
			1,
			IPP_OUTPUT_FORMAT_12_BIT_FIX);

	/* mpc TODO un-hardcode object ids
	 * for pseudo code pipe_move.c :
	 * add_plane_mpcc(added_plane_inst, mpcc_inst, ...);
	 * Do we want to cache the tree_cfg?
	 */

	/* TODO: build stream pipes group id. For now, use stream otg
	 * id as pipe group id
	 */
	pipe_ctx->scl_data.lb_params.alpha_en = per_pixel_alpha;
	pipe_ctx->mpc_idx = pipe_ctx->tg->inst;
	tree_cfg = &context->res_ctx.mpc_tree[pipe_ctx->mpc_idx];
	if (tree_cfg->num_pipes == 0) {
		tree_cfg->opp_id = pipe_ctx->tg->inst;
		for (i = 0; i < MAX_PIPES; i++) {
			tree_cfg->dpp[i] = 0xf;
			tree_cfg->mpcc[i] = 0xf;
		}
	}

	for (temp_pipe = pipe_ctx->top_pipe;
			temp_pipe != NULL; temp_pipe = temp_pipe->top_pipe)
		tree_pos++;

	tree_cfg->dpp[tree_pos] = pipe_ctx->pipe_idx;
	tree_cfg->mpcc[tree_pos] = pipe_ctx->pipe_idx;
	tree_cfg->per_pixel_alpha[tree_pos] = per_pixel_alpha;
	tree_cfg->num_pipes = tree_pos + 1;
	dcn10_set_mpc_tree(mpc, tree_cfg);
	for (top_pipe = pipe_ctx; top_pipe != NULL; top_pipe = top_pipe->top_pipe)
		mpcc_cfg.opp_id = top_pipe->opp->inst;
	mpcc_cfg.top_dpp_id = pipe_ctx->pipe_idx;
	if (pipe_ctx->bottom_pipe)
		mpcc_cfg.bot_mpcc_id = pipe_ctx->bottom_pipe->mpcc->inst;
	else
		mpcc_cfg.bot_mpcc_id = 0xf;
	mpcc_cfg.top_of_tree = !pipe_ctx->top_pipe;
	mpcc_cfg.per_pixel_alpha = per_pixel_alpha;
	if (!dc->current_context->res_ctx.pipe_ctx[pipe_ctx->pipe_idx].surface)
		pipe_ctx->mpcc->funcs->wait_for_idle(pipe_ctx->mpcc);
	pipe_ctx->mpcc->funcs->set(pipe_ctx->mpcc, &mpcc_cfg);

	color_space = pipe_ctx->stream->public.output_color_space;
	color_space_to_black_color(dc, color_space, &black_color);
	dcn10_set_mpc_background_color(mpc, pipe_ctx->pipe_idx, &black_color);
	color_space_to_black_color(
		dc, pipe_ctx->stream->public.output_color_space, &black_color);
	pipe_ctx->mpcc->funcs->set_bg_color(pipe_ctx->mpcc, &black_color);

	pipe_ctx->scl_data.lb_params.depth = LB_PIXEL_DEPTH_30BPP;
	/* scaler configuration */
@@ -1853,14 +1830,9 @@ static void dcn10_apply_ctx_for_surface(


		/* looking for top pipe to program */
		if (!pipe_ctx->top_pipe) {
			memcpy(context->res_ctx.mpc_tree,
					dc->current_context->res_ctx.mpc_tree,
					sizeof(struct mpc_tree_cfg) * dc->res_pool->pipe_count);

		if (!pipe_ctx->top_pipe)
			program_all_pipe_in_tree(dc, pipe_ctx, context);
	}
	}

	dm_logger_write(dc->ctx->logger, LOG_BANDWIDTH_CALCS,
			"\n============== Watermark parameters ==============\n"
@@ -1927,8 +1899,7 @@ static void dcn10_apply_ctx_for_surface(

		if ((!pipe_ctx->surface && old_pipe_ctx->surface)
				|| (!pipe_ctx->stream && old_pipe_ctx->stream))
			reset_front_end_for_pipe(dc,
					old_pipe_ctx, dc->current_context);
			reset_front_end_for_pipe(dc, old_pipe_ctx, dc->current_context);
	}
}

+70 −266
Original line number Diff line number Diff line
@@ -27,34 +27,26 @@
#include "dcn10_mpc.h"

#define REG(reg)\
	mpc->mpc_regs->reg
	mpcc10->mpcc_regs->reg

#define CTX \
	mpc->base.ctx
	mpcc10->base.ctx

#undef FN
#define FN(reg_name, field_name) \
	mpc->mpc_shift->field_name, mpc->mpc_mask->field_name
	mpcc10->mpcc_shift->field_name, mpcc10->mpcc_mask->field_name

#define MODE_TOP_ONLY 1
#define MODE_BLEND 3
#define BLND_PP_ALPHA 0
#define BLND_GLOBAL_ALPHA 2

/* Internal function to set mpc output mux */
static void set_output_mux(struct dcn10_mpc *mpc,
	uint8_t opp_id,
	uint8_t mpcc_id)
{
	if (mpcc_id != 0xf)
		REG_UPDATE(OPP_PIPE_CONTROL[opp_id],
				OPP_PIPE_CLOCK_EN, 1);

	REG_SET(MUX[opp_id], 0, MPC_OUT_MUX, mpcc_id);
}

void dcn10_set_mpc_background_color(struct dcn10_mpc *mpc,
	unsigned int mpcc_inst,
void dcn10_mpcc_set_bg_color(
		struct mpcc *mpcc,
		struct tg_color *bg_color)
{
	struct dcn10_mpcc *mpcc10 = TO_DCN10_MPCC(mpcc);
	/* mpc color is 12 bit.  tg_color is 10 bit */
	/* todo: might want to use 16 bit to represent color and have each
	 * hw block translate to correct color depth.
@@ -63,277 +55,89 @@ void dcn10_set_mpc_background_color(struct dcn10_mpc *mpc,
	uint32_t bg_g_y = bg_color->color_g_y << 2;
	uint32_t bg_b_cb = bg_color->color_b_cb << 2;

	REG_SET(MPCC_BG_R_CR[mpcc_inst], 0,
	REG_SET(MPCC_BG_R_CR, 0,
			MPCC_BG_R_CR, bg_r_cr);
	REG_SET(MPCC_BG_G_Y[mpcc_inst], 0,
	REG_SET(MPCC_BG_G_Y, 0,
			MPCC_BG_G_Y, bg_g_y);
	REG_SET(MPCC_BG_B_CB[mpcc_inst], 0,
	REG_SET(MPCC_BG_B_CB, 0,
			MPCC_BG_B_CB, bg_b_cb);
}

/* This function programs MPC tree configuration
 * Assume it is the initial time to setup MPC tree_configure, means
 * the instance of dpp/mpcc/opp specified in structure tree_cfg are
 * in idle status.
 * Before invoke this function, ensure that master lock of OPTC specified
 * by opp_id is set.
 *
 * tree_cfg[in] - new MPC_TREE_CFG
 */

void dcn10_set_mpc_tree(struct dcn10_mpc *mpc,
	struct mpc_tree_cfg *tree_cfg)
static void set_output_mux(struct dcn10_mpcc *mpcc10, int opp_id, int mpcc_id)
{
	int i;

	for (i = 0; i < tree_cfg->num_pipes; i++) {
		uint8_t mpcc_inst = tree_cfg->mpcc[i];

		REG_SET(MPCC_OPP_ID[mpcc_inst], 0,
			MPCC_OPP_ID, tree_cfg->opp_id);

		REG_SET(MPCC_TOP_SEL[mpcc_inst], 0,
			MPCC_TOP_SEL, tree_cfg->dpp[i]);

		if (i == tree_cfg->num_pipes-1) {
			REG_SET(MPCC_BOT_SEL[mpcc_inst], 0,
				MPCC_BOT_SEL, 0xF);

			REG_UPDATE(MPCC_CONTROL[mpcc_inst],
					MPCC_MODE, MODE_TOP_ONLY);
		} else {
			REG_SET(MPCC_BOT_SEL[mpcc_inst], 0,
				MPCC_BOT_SEL, tree_cfg->dpp[i+1]);

			REG_UPDATE(MPCC_CONTROL[mpcc_inst],
					MPCC_MODE, MODE_BLEND);
		}

		if (i == 0)
			set_output_mux(
				mpc, tree_cfg->opp_id, mpcc_inst);

		REG_UPDATE_2(MPCC_CONTROL[mpcc_inst],
				MPCC_ALPHA_BLND_MODE,
				tree_cfg->per_pixel_alpha[i] ? 0 : 2,
				MPCC_ALPHA_MULTIPLIED_MODE, 0);
	}
	ASSERT(mpcc10->opp_id == 0xf || opp_id == mpcc10->opp_id);
	mpcc10->opp_id = opp_id;
	REG_UPDATE(OPP_PIPE_CONTROL[opp_id], OPP_PIPE_CLOCK_EN, 1);
	REG_SET(MUX[opp_id], 0, MPC_OUT_MUX, mpcc_id);
}

/*
 * This is the function to remove current MPC tree specified by tree_cfg
 * Before invoke this function, ensure that master lock of OPTC specified
 * by opp_id is set.
 *
 *tree_cfg[in/out] - current MPC_TREE_CFG
 */
void dcn10_delete_mpc_tree(struct dcn10_mpc *mpc,
	struct mpc_tree_cfg *tree_cfg)
static void reset_output_mux(struct dcn10_mpcc *mpcc10)
{
	int i;

	for (i = 0; i < tree_cfg->num_pipes; i++) {
		uint8_t mpcc_inst = tree_cfg->mpcc[i];

		REG_SET(MPCC_OPP_ID[mpcc_inst], 0,
			MPCC_OPP_ID, 0xf);

		REG_SET(MPCC_TOP_SEL[mpcc_inst], 0,
			MPCC_TOP_SEL, 0xf);

		REG_SET(MPCC_BOT_SEL[mpcc_inst], 0,
			MPCC_BOT_SEL, 0xF);

		/* add remove dpp/mpcc pair into pending list
		 * TODO FPGA AddToPendingList if empty from pseudo code
		 */
		tree_cfg->dpp[i] = 0xf;
		tree_cfg->mpcc[i] = 0xf;
		tree_cfg->per_pixel_alpha[i] = false;
	}
	set_output_mux(mpc, tree_cfg->opp_id, 0xf);
	tree_cfg->opp_id = 0xf;
	tree_cfg->num_pipes = 0;
	REG_SET(MUX[mpcc10->opp_id], 0, MPC_OUT_MUX, 0xf);
	REG_UPDATE(OPP_PIPE_CONTROL[mpcc10->opp_id], OPP_PIPE_CLOCK_EN, 0);
	mpcc10->opp_id = 0xf;
}

/* TODO FPGA: how to handle DPP?
 * Function to remove one of pipe from MPC configure tree by dpp idx
 * Before invoke this function, ensure that master lock of OPTC specified
 * by opp_id is set
 * This function can be invoke multiple times to remove more than 1 dpps.
 *
 * tree_cfg[in/out] - current MPC_TREE_CFG
 * idx[in] - index of dpp from tree_cfg to be removed.
 */
bool dcn10_remove_dpp(struct dcn10_mpc *mpc,
	struct mpc_tree_cfg *tree_cfg,
	uint8_t idx)
static void dcn10_mpcc_set(struct mpcc *mpcc, struct mpcc_cfg *cfg)
{
	int i;
	uint8_t mpcc_inst;
	bool found = false;

	/* find dpp_idx from dpp array of tree_cfg */
	for (i = 0; i < tree_cfg->num_pipes; i++) {
		if (tree_cfg->dpp[i] == idx) {
			found = true;
			break;
		}
	}

	if (!found) {
		BREAK_TO_DEBUGGER();
		return false;
	}
	mpcc_inst = tree_cfg->mpcc[i];

	REG_SET(MPCC_OPP_ID[mpcc_inst], 0,
			MPCC_OPP_ID, 0xf);

	REG_SET(MPCC_TOP_SEL[mpcc_inst], 0,
			MPCC_TOP_SEL, 0xf);

	REG_SET(MPCC_BOT_SEL[mpcc_inst], 0,
			MPCC_BOT_SEL, 0xf);

	if (i == 0) {
		if (tree_cfg->num_pipes > 1)
			set_output_mux(mpc,
				tree_cfg->opp_id, tree_cfg->mpcc[i+1]);
	struct dcn10_mpcc *mpcc10 = TO_DCN10_MPCC(mpcc);
	int alpha_blnd_mode = cfg->per_pixel_alpha ?
			BLND_PP_ALPHA : BLND_GLOBAL_ALPHA;
	int mpcc_mode = cfg->bot_mpcc_id != 0xf ?
				MODE_BLEND : MODE_TOP_ONLY;

	REG_SET(MPCC_OPP_ID, 0,
		MPCC_OPP_ID, cfg->opp_id);

	REG_SET(MPCC_TOP_SEL, 0,
		MPCC_TOP_SEL, cfg->top_dpp_id);

	REG_SET(MPCC_BOT_SEL, 0,
		MPCC_BOT_SEL, cfg->bot_mpcc_id);

	REG_SET_4(MPCC_CONTROL, 0xffffffff,
		MPCC_MODE, mpcc_mode,
		MPCC_ALPHA_BLND_MODE, alpha_blnd_mode,
		MPCC_ALPHA_MULTIPLIED_MODE, 0/*TODO: cfg->per_pixel_alpha*/,
		MPCC_BLND_ACTIVE_OVERLAP_ONLY, cfg->top_of_tree);

	if (cfg->top_of_tree) {
		if (cfg->opp_id != 0xf)
			set_output_mux(mpcc10, cfg->opp_id, mpcc->inst);
		else
			set_output_mux(mpc, tree_cfg->opp_id, 0xf);
	} else if (i == tree_cfg->num_pipes-1) {
		mpcc_inst = tree_cfg->mpcc[i - 1];

		REG_SET(MPCC_BOT_SEL[mpcc_inst], 0,
				MPCC_BOT_SEL, 0xF);

		/* prev mpc is now last, set to top only*/
		REG_UPDATE(MPCC_CONTROL[mpcc_inst],
				MPCC_MODE, MODE_TOP_ONLY);
	} else {
		mpcc_inst = tree_cfg->mpcc[i - 1];

		REG_SET(MPCC_BOT_SEL[mpcc_inst], 0,
			MPCC_BOT_SEL, tree_cfg->mpcc[i+1]);
	}

	/* update tree_cfg structure */
	while (i < tree_cfg->num_pipes - 1) {
		tree_cfg->dpp[i] = tree_cfg->dpp[i+1];
		tree_cfg->mpcc[i] = tree_cfg->mpcc[i+1];
		tree_cfg->per_pixel_alpha[i] = tree_cfg->per_pixel_alpha[i+1];
		i++;
			reset_output_mux(mpcc10);
	}
	tree_cfg->num_pipes--;

	return true;
}

/* TODO FPGA: how to handle DPP?
 * Function to add DPP/MPCC pair into MPC configure tree by position.
 * Before invoke this function, ensure that master lock of OPTC specified
 * by opp_id is set
 * This function can be invoke multiple times to add more than 1 pipes.
 *
 * tree_cfg[in/out] - current MPC_TREE_CFG
 * dpp_idx[in]	 - index of an idle dpp insatnce to be added.
 * mpcc_idx[in]	 - index of an idle mpcc instance to be added.
 * poistion[in]	 - position of dpp/mpcc pair to be added into current tree_cfg
 *                 0 means insert to the most top layer of MPC tree
 */
void dcn10_add_dpp(struct dcn10_mpc *mpc,
	struct mpc_tree_cfg *tree_cfg,
	uint8_t dpp_idx,
	uint8_t mpcc_idx,
	uint8_t per_pixel_alpha,
	uint8_t position)
static void dcn10_mpcc_wait_idle(struct mpcc *mpcc)
{
	uint8_t prev;
	uint8_t next;

	REG_SET(MPCC_OPP_ID[mpcc_idx], 0,
			MPCC_OPP_ID, tree_cfg->opp_id);
	REG_SET(MPCC_TOP_SEL[mpcc_idx], 0,
			MPCC_TOP_SEL, dpp_idx);

	if (position == 0) {
		/* idle dpp/mpcc is added to the top layer of tree */
		REG_SET(MPCC_BOT_SEL[mpcc_idx], 0,
				MPCC_BOT_SEL, tree_cfg->mpcc[0]);
	struct dcn10_mpcc *mpcc10 = TO_DCN10_MPCC(mpcc);

		/* bottom mpc is always top only */
		REG_UPDATE(MPCC_CONTROL[mpcc_idx],
				MPCC_MODE, MODE_TOP_ONLY);
		/* opp will get new output. from new added mpcc */
		set_output_mux(mpc, tree_cfg->opp_id, mpcc_idx);

	} else if (position == tree_cfg->num_pipes) {
		/* idle dpp/mpcc is added to the bottom layer of tree */

		/* get instance of previous bottom mpcc, set to middle layer */
		prev = tree_cfg->mpcc[position - 1];

		REG_SET(MPCC_BOT_SEL[prev], 0,
				MPCC_BOT_SEL, mpcc_idx);

		/* all mpcs other than bottom need to blend */
		REG_UPDATE(MPCC_CONTROL[prev],
				MPCC_MODE, MODE_BLEND);

		/* mpcc_idx become new bottom mpcc*/
		REG_SET(MPCC_BOT_SEL[mpcc_idx], 0,
				MPCC_BOT_SEL, 0xf);

		/* bottom mpc is always top only */
		REG_UPDATE(MPCC_CONTROL[mpcc_idx],
				MPCC_MODE, MODE_TOP_ONLY);
	} else {
		/* idle dpp/mpcc is added to middle of tree */
		prev = tree_cfg->mpcc[position - 1]; /* mpc a */
		next = tree_cfg->mpcc[position]; /* mpc b */

		/* connect mpc inserted below mpc a*/
		REG_SET(MPCC_BOT_SEL[prev], 0,
				MPCC_BOT_SEL, mpcc_idx);
	REG_WAIT(MPCC_STATUS, MPCC_IDLE, 1, 1000, 1000);
}

		/* blend on mpc being inserted */
		REG_UPDATE(MPCC_CONTROL[mpcc_idx],
				MPCC_MODE, MODE_BLEND);

		/* Connect mpc b below one inserted */
		REG_SET(MPCC_BOT_SEL[mpcc_idx], 0,
				MPCC_BOT_SEL, next);
const struct mpcc_funcs dcn10_mpcc_funcs = {
		.set = dcn10_mpcc_set,
		.wait_for_idle = dcn10_mpcc_wait_idle,
		.set_bg_color = dcn10_mpcc_set_bg_color,
};

	}
	/* premultiplied mode only if alpha is on for the layer*/
	REG_UPDATE_2(MPCC_CONTROL[mpcc_idx],
			MPCC_ALPHA_BLND_MODE,
			tree_cfg->per_pixel_alpha[position] ? 0 : 2,
			MPCC_ALPHA_MULTIPLIED_MODE, 0);
void dcn10_mpcc_construct(struct dcn10_mpcc *mpcc10,
	struct dc_context *ctx,
	const struct dcn_mpcc_registers *mpcc_regs,
	const struct dcn_mpcc_shift *mpcc_shift,
	const struct dcn_mpcc_mask *mpcc_mask,
	int inst)
{
	mpcc10->base.ctx = ctx;

	/*
	 * iterating from the last mpc/dpp pair to the one being added, shift
	 * them down one position
	 */
	for (next = tree_cfg->num_pipes; next > position; next--) {
		tree_cfg->dpp[next] = tree_cfg->dpp[next - 1];
		tree_cfg->mpcc[next] = tree_cfg->mpcc[next - 1];
		tree_cfg->per_pixel_alpha[next] = tree_cfg->per_pixel_alpha[next - 1];
	}
	mpcc10->base.inst = inst;
	mpcc10->base.funcs = &dcn10_mpcc_funcs;

	/* insert the new mpc/dpp pair into the tree_cfg*/
	tree_cfg->dpp[position] = dpp_idx;
	tree_cfg->mpcc[position] = mpcc_idx;
	tree_cfg->per_pixel_alpha[position] = per_pixel_alpha;
	tree_cfg->num_pipes++;
}
	mpcc10->mpcc_regs = mpcc_regs;
	mpcc10->mpcc_shift = mpcc_shift;
	mpcc10->mpcc_mask = mpcc_mask;

void wait_mpcc_idle(struct dcn10_mpc *mpc,
	uint8_t mpcc_id)
{
	REG_WAIT(MPCC_STATUS[mpcc_id],
			MPCC_IDLE, 1,
			1000, 1000);
	mpcc10->opp_id = inst;
}
Loading