}
 }
 
+static int get_num_of_free_pipes(const struct resource_pool *pool, const struct dc_state *context)
+{
+       int i;
+       int count = 0;
+
+       for (i = 0; i < pool->pipe_count; i++)
+               if (resource_is_pipe_type(&context->res_ctx.pipe_ctx[i], FREE_PIPE))
+                       count++;
+       return count;
+}
+
 enum dc_status resource_add_otg_master_for_stream_output(struct dc_state *new_ctx,
                const struct resource_pool *pool,
                struct dc_stream_state *stream)
                struct dc_state *cur_ctx,
                struct resource_pool *pool)
 {
-       struct pipe_ctx *opp_head_pipe, *sec_pipe, *tail_pipe;
+       struct pipe_ctx *sec_pipe, *tail_pipe;
+       struct pipe_ctx *opp_heads[MAX_PIPES];
+       int opp_head_count;
+       int i;
 
        if (!pool->funcs->acquire_free_pipe_as_secondary_dpp_pipe) {
                ASSERT(0);
                return false;
        }
 
-       opp_head_pipe = otg_master_pipe;
-       while (opp_head_pipe) {
+       opp_head_count = resource_get_opp_heads_for_otg_master(otg_master_pipe,
+                       &new_ctx->res_ctx, opp_heads);
+       if (get_num_of_free_pipes(pool, new_ctx) < opp_head_count)
+               /* not enough free pipes */
+               return false;
+
+       for (i = 0; i < opp_head_count; i++) {
                sec_pipe = pool->funcs->acquire_free_pipe_as_secondary_dpp_pipe(
                                cur_ctx,
                                new_ctx,
                                pool,
-                               opp_head_pipe);
-               if (!sec_pipe) {
-                       /* try tearing down MPCC combine */
-                       int pipe_idx = acquire_first_split_pipe(
-                                       &new_ctx->res_ctx, pool,
-                                       otg_master_pipe->stream);
-
-                       if (pipe_idx >= 0)
-                               sec_pipe = &new_ctx->res_ctx.pipe_ctx[pipe_idx];
-               }
-
-               if (!sec_pipe)
-                       return false;
-
+                               opp_heads[i]);
+               ASSERT(sec_pipe);
                sec_pipe->plane_state = plane_state;
 
                /* establish pipe relationship */
-               tail_pipe = get_tail_pipe(opp_head_pipe);
+               tail_pipe = get_tail_pipe(opp_heads[i]);
                tail_pipe->bottom_pipe = sec_pipe;
                sec_pipe->top_pipe = tail_pipe;
                sec_pipe->bottom_pipe = NULL;
                } else {
                        sec_pipe->prev_odm_pipe = NULL;
                }
-
-               opp_head_pipe = opp_head_pipe->next_odm_pipe;
        }
        return true;
 }
 
        return DC_OK;
 }
 
+static void remove_mpc_combine_for_stream(const struct dc *dc,
+               struct dc_state *new_ctx,
+               const struct dc_state *cur_ctx,
+               struct dc_stream_status *status)
+{
+       int i;
+
+       for (i = 0; i < status->plane_count; i++)
+               resource_update_pipes_for_plane_with_slice_count(
+                               new_ctx, cur_ctx, dc->res_pool,
+                               status->plane_states[i], 1);
+}
+
 bool dc_state_add_plane(
                const struct dc *dc,
                struct dc_stream_state *stream,
        struct pipe_ctx *otg_master_pipe;
        struct dc_stream_status *stream_status = NULL;
        bool added = false;
+       int odm_slice_count;
+       int i;
 
        stream_status = dc_state_get_stream_status(state, stream);
+       otg_master_pipe = resource_get_otg_master_for_stream(
+                       &state->res_ctx, stream);
        if (stream_status == NULL) {
                dm_error("Existing stream not found; failed to attach surface!\n");
                goto out;
                dm_error("Surface: can not attach plane_state %p! Maximum is: %d\n",
                                plane_state, MAX_SURFACE_NUM);
                goto out;
+       } else if (!otg_master_pipe) {
+               goto out;
        }
 
-       if (stream_status->plane_count == 0 && dc->config.enable_windowed_mpo_odm)
-               /* ODM combine could prevent us from supporting more planes
-                * we will reset ODM slice count back to 1 when all planes have
-                * been removed to maximize the amount of planes supported when
-                * new planes are added.
-                */
-               resource_update_pipes_for_stream_with_slice_count(
-                               state, dc->current_state, dc->res_pool, stream, 1);
+       added = resource_append_dpp_pipes_for_plane_composition(state,
+                       dc->current_state, pool, otg_master_pipe, plane_state);
 
-       otg_master_pipe = resource_get_otg_master_for_stream(
-                       &state->res_ctx, stream);
-       if (otg_master_pipe)
+       if (!added) {
+               /* try to remove MPC combine to free up pipes */
+               for (i = 0; i < state->stream_count; i++)
+                       remove_mpc_combine_for_stream(dc, state,
+                                       dc->current_state,
+                                       &state->stream_status[i]);
                added = resource_append_dpp_pipes_for_plane_composition(state,
-                               dc->current_state, pool, otg_master_pipe, plane_state);
+                                       dc->current_state, pool,
+                                       otg_master_pipe, plane_state);
+       }
+
+       if (!added) {
+               /* try to decrease ODM slice count gradually to free up pipes */
+               odm_slice_count = resource_get_odm_slice_count(otg_master_pipe);
+               for (i = odm_slice_count - 1; i > 0; i--) {
+                       resource_update_pipes_for_stream_with_slice_count(state,
+                                       dc->current_state, dc->res_pool, stream,
+                                       i);
+                       added = resource_append_dpp_pipes_for_plane_composition(
+                                       state,
+                                       dc->current_state, pool,
+                                       otg_master_pipe, plane_state);
+                       if (added)
+                               break;
+               }
+       }
 
        if (added) {
                stream_status->plane_states[stream_status->plane_count] =
 
        stream_status->plane_states[stream_status->plane_count] = NULL;
 
-       if (stream_status->plane_count == 0 && dc->config.enable_windowed_mpo_odm)
-               /* ODM combine could prevent us from supporting more planes
-                * we will reset ODM slice count back to 1 when all planes have
-                * been removed to maximize the amount of planes supported when
-                * new planes are added.
-                */
-               resource_update_pipes_for_stream_with_slice_count(
-                               state, dc->current_state, dc->res_pool, stream, 1);
-
        return true;
 }