xref: /openbmc/linux/drivers/gpu/drm/msm/disp/dpu1/dpu_rm.c (revision a266ef69b890f099069cf51bb40572611c435a54)
1 // SPDX-License-Identifier: GPL-2.0-only
2 /*
3  * Copyright (c) 2016-2018, The Linux Foundation. All rights reserved.
4  */
5 
6 #define pr_fmt(fmt)	"[drm:%s] " fmt, __func__
7 #include "dpu_kms.h"
8 #include "dpu_hw_lm.h"
9 #include "dpu_hw_ctl.h"
10 #include "dpu_hw_pingpong.h"
11 #include "dpu_hw_intf.h"
12 #include "dpu_hw_wb.h"
13 #include "dpu_hw_dspp.h"
14 #include "dpu_hw_merge3d.h"
15 #include "dpu_hw_dsc.h"
16 #include "dpu_encoder.h"
17 #include "dpu_trace.h"
18 
19 
20 static inline bool reserved_by_other(uint32_t *res_map, int idx,
21 				     uint32_t enc_id)
22 {
23 	return res_map[idx] && res_map[idx] != enc_id;
24 }
25 
26 /**
27  * struct dpu_rm_requirements - Reservation requirements parameter bundle
28  * @topology:  selected topology for the display
29  * @hw_res:	   Hardware resources required as reported by the encoders
30  */
31 struct dpu_rm_requirements {
32 	struct msm_display_topology topology;
33 };
34 
35 int dpu_rm_destroy(struct dpu_rm *rm)
36 {
37 	int i;
38 
39 	for (i = 0; i < ARRAY_SIZE(rm->dspp_blks); i++) {
40 		struct dpu_hw_dspp *hw;
41 
42 		if (rm->dspp_blks[i]) {
43 			hw = to_dpu_hw_dspp(rm->dspp_blks[i]);
44 			dpu_hw_dspp_destroy(hw);
45 		}
46 	}
47 	for (i = 0; i < ARRAY_SIZE(rm->pingpong_blks); i++) {
48 		struct dpu_hw_pingpong *hw;
49 
50 		if (rm->pingpong_blks[i]) {
51 			hw = to_dpu_hw_pingpong(rm->pingpong_blks[i]);
52 			dpu_hw_pingpong_destroy(hw);
53 		}
54 	}
55 	for (i = 0; i < ARRAY_SIZE(rm->merge_3d_blks); i++) {
56 		struct dpu_hw_merge_3d *hw;
57 
58 		if (rm->merge_3d_blks[i]) {
59 			hw = to_dpu_hw_merge_3d(rm->merge_3d_blks[i]);
60 			dpu_hw_merge_3d_destroy(hw);
61 		}
62 	}
63 	for (i = 0; i < ARRAY_SIZE(rm->mixer_blks); i++) {
64 		struct dpu_hw_mixer *hw;
65 
66 		if (rm->mixer_blks[i]) {
67 			hw = to_dpu_hw_mixer(rm->mixer_blks[i]);
68 			dpu_hw_lm_destroy(hw);
69 		}
70 	}
71 	for (i = 0; i < ARRAY_SIZE(rm->ctl_blks); i++) {
72 		struct dpu_hw_ctl *hw;
73 
74 		if (rm->ctl_blks[i]) {
75 			hw = to_dpu_hw_ctl(rm->ctl_blks[i]);
76 			dpu_hw_ctl_destroy(hw);
77 		}
78 	}
79 	for (i = 0; i < ARRAY_SIZE(rm->hw_intf); i++)
80 		dpu_hw_intf_destroy(rm->hw_intf[i]);
81 
82 	for (i = 0; i < ARRAY_SIZE(rm->dsc_blks); i++) {
83 		struct dpu_hw_dsc *hw;
84 
85 		if (rm->dsc_blks[i]) {
86 			hw = to_dpu_hw_dsc(rm->dsc_blks[i]);
87 			dpu_hw_dsc_destroy(hw);
88 		}
89 	}
90 
91 	for (i = 0; i < ARRAY_SIZE(rm->hw_wb); i++)
92 		dpu_hw_wb_destroy(rm->hw_wb[i]);
93 
94 	return 0;
95 }
96 
97 int dpu_rm_init(struct dpu_rm *rm,
98 		const struct dpu_mdss_cfg *cat,
99 		void __iomem *mmio)
100 {
101 	int rc, i;
102 
103 	if (!rm || !cat || !mmio) {
104 		DPU_ERROR("invalid kms\n");
105 		return -EINVAL;
106 	}
107 
108 	/* Clear, setup lists */
109 	memset(rm, 0, sizeof(*rm));
110 
111 	/* Interrogate HW catalog and create tracking items for hw blocks */
112 	for (i = 0; i < cat->mixer_count; i++) {
113 		struct dpu_hw_mixer *hw;
114 		const struct dpu_lm_cfg *lm = &cat->mixer[i];
115 
116 		if (lm->pingpong == PINGPONG_MAX) {
117 			DPU_DEBUG("skip mixer %d without pingpong\n", lm->id);
118 			continue;
119 		}
120 
121 		if (lm->id < LM_0 || lm->id >= LM_MAX) {
122 			DPU_ERROR("skip mixer %d with invalid id\n", lm->id);
123 			continue;
124 		}
125 		hw = dpu_hw_lm_init(lm->id, mmio, cat);
126 		if (IS_ERR(hw)) {
127 			rc = PTR_ERR(hw);
128 			DPU_ERROR("failed lm object creation: err %d\n", rc);
129 			goto fail;
130 		}
131 		rm->mixer_blks[lm->id - LM_0] = &hw->base;
132 	}
133 
134 	for (i = 0; i < cat->merge_3d_count; i++) {
135 		struct dpu_hw_merge_3d *hw;
136 		const struct dpu_merge_3d_cfg *merge_3d = &cat->merge_3d[i];
137 
138 		if (merge_3d->id < MERGE_3D_0 || merge_3d->id >= MERGE_3D_MAX) {
139 			DPU_ERROR("skip merge_3d %d with invalid id\n", merge_3d->id);
140 			continue;
141 		}
142 		hw = dpu_hw_merge_3d_init(merge_3d->id, mmio, cat);
143 		if (IS_ERR(hw)) {
144 			rc = PTR_ERR(hw);
145 			DPU_ERROR("failed merge_3d object creation: err %d\n",
146 				rc);
147 			goto fail;
148 		}
149 		rm->merge_3d_blks[merge_3d->id - MERGE_3D_0] = &hw->base;
150 	}
151 
152 	for (i = 0; i < cat->pingpong_count; i++) {
153 		struct dpu_hw_pingpong *hw;
154 		const struct dpu_pingpong_cfg *pp = &cat->pingpong[i];
155 
156 		if (pp->id < PINGPONG_0 || pp->id >= PINGPONG_MAX) {
157 			DPU_ERROR("skip pingpong %d with invalid id\n", pp->id);
158 			continue;
159 		}
160 		hw = dpu_hw_pingpong_init(pp->id, mmio, cat);
161 		if (IS_ERR(hw)) {
162 			rc = PTR_ERR(hw);
163 			DPU_ERROR("failed pingpong object creation: err %d\n",
164 				rc);
165 			goto fail;
166 		}
167 		if (pp->merge_3d && pp->merge_3d < MERGE_3D_MAX)
168 			hw->merge_3d = to_dpu_hw_merge_3d(rm->merge_3d_blks[pp->merge_3d - MERGE_3D_0]);
169 		rm->pingpong_blks[pp->id - PINGPONG_0] = &hw->base;
170 	}
171 
172 	for (i = 0; i < cat->intf_count; i++) {
173 		struct dpu_hw_intf *hw;
174 		const struct dpu_intf_cfg *intf = &cat->intf[i];
175 
176 		if (intf->type == INTF_NONE) {
177 			DPU_DEBUG("skip intf %d with type none\n", i);
178 			continue;
179 		}
180 		if (intf->id < INTF_0 || intf->id >= INTF_MAX) {
181 			DPU_ERROR("skip intf %d with invalid id\n", intf->id);
182 			continue;
183 		}
184 		hw = dpu_hw_intf_init(intf->id, mmio, cat);
185 		if (IS_ERR(hw)) {
186 			rc = PTR_ERR(hw);
187 			DPU_ERROR("failed intf object creation: err %d\n", rc);
188 			goto fail;
189 		}
190 		rm->hw_intf[intf->id - INTF_0] = hw;
191 	}
192 
193 	for (i = 0; i < cat->wb_count; i++) {
194 		struct dpu_hw_wb *hw;
195 		const struct dpu_wb_cfg *wb = &cat->wb[i];
196 
197 		if (wb->id < WB_0 || wb->id >= WB_MAX) {
198 			DPU_ERROR("skip intf %d with invalid id\n", wb->id);
199 			continue;
200 		}
201 
202 		hw = dpu_hw_wb_init(wb->id, mmio, cat);
203 		if (IS_ERR(hw)) {
204 			rc = PTR_ERR(hw);
205 			DPU_ERROR("failed wb object creation: err %d\n", rc);
206 			goto fail;
207 		}
208 		rm->hw_wb[wb->id - WB_0] = hw;
209 	}
210 
211 	for (i = 0; i < cat->ctl_count; i++) {
212 		struct dpu_hw_ctl *hw;
213 		const struct dpu_ctl_cfg *ctl = &cat->ctl[i];
214 
215 		if (ctl->id < CTL_0 || ctl->id >= CTL_MAX) {
216 			DPU_ERROR("skip ctl %d with invalid id\n", ctl->id);
217 			continue;
218 		}
219 		hw = dpu_hw_ctl_init(ctl->id, mmio, cat);
220 		if (IS_ERR(hw)) {
221 			rc = PTR_ERR(hw);
222 			DPU_ERROR("failed ctl object creation: err %d\n", rc);
223 			goto fail;
224 		}
225 		rm->ctl_blks[ctl->id - CTL_0] = &hw->base;
226 	}
227 
228 	for (i = 0; i < cat->dspp_count; i++) {
229 		struct dpu_hw_dspp *hw;
230 		const struct dpu_dspp_cfg *dspp = &cat->dspp[i];
231 
232 		if (dspp->id < DSPP_0 || dspp->id >= DSPP_MAX) {
233 			DPU_ERROR("skip dspp %d with invalid id\n", dspp->id);
234 			continue;
235 		}
236 		hw = dpu_hw_dspp_init(dspp->id, mmio, cat);
237 		if (IS_ERR(hw)) {
238 			rc = PTR_ERR(hw);
239 			DPU_ERROR("failed dspp object creation: err %d\n", rc);
240 			goto fail;
241 		}
242 		rm->dspp_blks[dspp->id - DSPP_0] = &hw->base;
243 	}
244 
245 	for (i = 0; i < cat->dsc_count; i++) {
246 		struct dpu_hw_dsc *hw;
247 		const struct dpu_dsc_cfg *dsc = &cat->dsc[i];
248 
249 		hw = dpu_hw_dsc_init(dsc->id, mmio, cat);
250 		if (IS_ERR_OR_NULL(hw)) {
251 			rc = PTR_ERR(hw);
252 			DPU_ERROR("failed dsc object creation: err %d\n", rc);
253 			goto fail;
254 		}
255 		rm->dsc_blks[dsc->id - DSC_0] = &hw->base;
256 	}
257 
258 	return 0;
259 
260 fail:
261 	dpu_rm_destroy(rm);
262 
263 	return rc ? rc : -EFAULT;
264 }
265 
266 static bool _dpu_rm_needs_split_display(const struct msm_display_topology *top)
267 {
268 	return top->num_intf > 1;
269 }
270 
271 /**
272  * _dpu_rm_check_lm_peer - check if a mixer is a peer of the primary
273  * @rm: dpu resource manager handle
274  * @primary_idx: index of primary mixer in rm->mixer_blks[]
275  * @peer_idx: index of other mixer in rm->mixer_blks[]
276  * Return: true if rm->mixer_blks[peer_idx] is a peer of
277  *          rm->mixer_blks[primary_idx]
278  */
279 static bool _dpu_rm_check_lm_peer(struct dpu_rm *rm, int primary_idx,
280 		int peer_idx)
281 {
282 	const struct dpu_lm_cfg *prim_lm_cfg;
283 	const struct dpu_lm_cfg *peer_cfg;
284 
285 	prim_lm_cfg = to_dpu_hw_mixer(rm->mixer_blks[primary_idx])->cap;
286 	peer_cfg = to_dpu_hw_mixer(rm->mixer_blks[peer_idx])->cap;
287 
288 	if (!test_bit(peer_cfg->id, &prim_lm_cfg->lm_pair_mask)) {
289 		DPU_DEBUG("lm %d not peer of lm %d\n", peer_cfg->id,
290 				peer_cfg->id);
291 		return false;
292 	}
293 	return true;
294 }
295 
296 /**
297  * _dpu_rm_check_lm_and_get_connected_blks - check if proposed layer mixer meets
298  *	proposed use case requirements, incl. hardwired dependent blocks like
299  *	pingpong
300  * @rm: dpu resource manager handle
301  * @global_state: resources shared across multiple kms objects
302  * @enc_id: encoder id requesting for allocation
303  * @lm_idx: index of proposed layer mixer in rm->mixer_blks[], function checks
304  *      if lm, and all other hardwired blocks connected to the lm (pp) is
305  *      available and appropriate
306  * @pp_idx: output parameter, index of pingpong block attached to the layer
307  *      mixer in rm->pingpong_blks[].
308  * @dspp_idx: output parameter, index of dspp block attached to the layer
309  *      mixer in rm->dspp_blks[].
310  * @reqs: input parameter, rm requirements for HW blocks needed in the
311  *      datapath.
312  * Return: true if lm matches all requirements, false otherwise
313  */
314 static bool _dpu_rm_check_lm_and_get_connected_blks(struct dpu_rm *rm,
315 		struct dpu_global_state *global_state,
316 		uint32_t enc_id, int lm_idx, int *pp_idx, int *dspp_idx,
317 		struct dpu_rm_requirements *reqs)
318 {
319 	const struct dpu_lm_cfg *lm_cfg;
320 	int idx;
321 
322 	/* Already reserved? */
323 	if (reserved_by_other(global_state->mixer_to_enc_id, lm_idx, enc_id)) {
324 		DPU_DEBUG("lm %d already reserved\n", lm_idx + LM_0);
325 		return false;
326 	}
327 
328 	lm_cfg = to_dpu_hw_mixer(rm->mixer_blks[lm_idx])->cap;
329 	idx = lm_cfg->pingpong - PINGPONG_0;
330 	if (idx < 0 || idx >= ARRAY_SIZE(rm->pingpong_blks)) {
331 		DPU_ERROR("failed to get pp on lm %d\n", lm_cfg->pingpong);
332 		return false;
333 	}
334 
335 	if (reserved_by_other(global_state->pingpong_to_enc_id, idx, enc_id)) {
336 		DPU_DEBUG("lm %d pp %d already reserved\n", lm_cfg->id,
337 				lm_cfg->pingpong);
338 		return false;
339 	}
340 	*pp_idx = idx;
341 
342 	if (!reqs->topology.num_dspp)
343 		return true;
344 
345 	idx = lm_cfg->dspp - DSPP_0;
346 	if (idx < 0 || idx >= ARRAY_SIZE(rm->dspp_blks)) {
347 		DPU_ERROR("failed to get dspp on lm %d\n", lm_cfg->dspp);
348 		return false;
349 	}
350 
351 	if (reserved_by_other(global_state->dspp_to_enc_id, idx, enc_id)) {
352 		DPU_DEBUG("lm %d dspp %d already reserved\n", lm_cfg->id,
353 				lm_cfg->dspp);
354 		return false;
355 	}
356 	*dspp_idx = idx;
357 
358 	return true;
359 }
360 
361 static int _dpu_rm_reserve_lms(struct dpu_rm *rm,
362 			       struct dpu_global_state *global_state,
363 			       uint32_t enc_id,
364 			       struct dpu_rm_requirements *reqs)
365 
366 {
367 	int lm_idx[MAX_BLOCKS];
368 	int pp_idx[MAX_BLOCKS];
369 	int dspp_idx[MAX_BLOCKS] = {0};
370 	int i, j, lm_count = 0;
371 
372 	if (!reqs->topology.num_lm) {
373 		DPU_ERROR("invalid number of lm: %d\n", reqs->topology.num_lm);
374 		return -EINVAL;
375 	}
376 
377 	/* Find a primary mixer */
378 	for (i = 0; i < ARRAY_SIZE(rm->mixer_blks) &&
379 			lm_count < reqs->topology.num_lm; i++) {
380 		if (!rm->mixer_blks[i])
381 			continue;
382 
383 		lm_count = 0;
384 		lm_idx[lm_count] = i;
385 
386 		if (!_dpu_rm_check_lm_and_get_connected_blks(rm, global_state,
387 				enc_id, i, &pp_idx[lm_count],
388 				&dspp_idx[lm_count], reqs)) {
389 			continue;
390 		}
391 
392 		++lm_count;
393 
394 		/* Valid primary mixer found, find matching peers */
395 		for (j = i + 1; j < ARRAY_SIZE(rm->mixer_blks) &&
396 				lm_count < reqs->topology.num_lm; j++) {
397 			if (!rm->mixer_blks[j])
398 				continue;
399 
400 			if (!_dpu_rm_check_lm_peer(rm, i, j)) {
401 				DPU_DEBUG("lm %d not peer of lm %d\n", LM_0 + j,
402 						LM_0 + i);
403 				continue;
404 			}
405 
406 			if (!_dpu_rm_check_lm_and_get_connected_blks(rm,
407 					global_state, enc_id, j,
408 					&pp_idx[lm_count], &dspp_idx[lm_count],
409 					reqs)) {
410 				continue;
411 			}
412 
413 			lm_idx[lm_count] = j;
414 			++lm_count;
415 		}
416 	}
417 
418 	if (lm_count != reqs->topology.num_lm) {
419 		DPU_DEBUG("unable to find appropriate mixers\n");
420 		return -ENAVAIL;
421 	}
422 
423 	for (i = 0; i < lm_count; i++) {
424 		global_state->mixer_to_enc_id[lm_idx[i]] = enc_id;
425 		global_state->pingpong_to_enc_id[pp_idx[i]] = enc_id;
426 		global_state->dspp_to_enc_id[dspp_idx[i]] =
427 			reqs->topology.num_dspp ? enc_id : 0;
428 
429 		trace_dpu_rm_reserve_lms(lm_idx[i] + LM_0, enc_id,
430 					 pp_idx[i] + PINGPONG_0);
431 	}
432 
433 	return 0;
434 }
435 
436 static int _dpu_rm_reserve_ctls(
437 		struct dpu_rm *rm,
438 		struct dpu_global_state *global_state,
439 		uint32_t enc_id,
440 		const struct msm_display_topology *top)
441 {
442 	int ctl_idx[MAX_BLOCKS];
443 	int i = 0, j, num_ctls;
444 	bool needs_split_display;
445 
446 	/* each hw_intf needs its own hw_ctrl to program its control path */
447 	num_ctls = top->num_intf;
448 
449 	needs_split_display = _dpu_rm_needs_split_display(top);
450 
451 	for (j = 0; j < ARRAY_SIZE(rm->ctl_blks); j++) {
452 		const struct dpu_hw_ctl *ctl;
453 		unsigned long features;
454 		bool has_split_display;
455 
456 		if (!rm->ctl_blks[j])
457 			continue;
458 		if (reserved_by_other(global_state->ctl_to_enc_id, j, enc_id))
459 			continue;
460 
461 		ctl = to_dpu_hw_ctl(rm->ctl_blks[j]);
462 		features = ctl->caps->features;
463 		has_split_display = BIT(DPU_CTL_SPLIT_DISPLAY) & features;
464 
465 		DPU_DEBUG("ctl %d caps 0x%lX\n", j + CTL_0, features);
466 
467 		if (needs_split_display != has_split_display)
468 			continue;
469 
470 		ctl_idx[i] = j;
471 		DPU_DEBUG("ctl %d match\n", j + CTL_0);
472 
473 		if (++i == num_ctls)
474 			break;
475 
476 	}
477 
478 	if (i != num_ctls)
479 		return -ENAVAIL;
480 
481 	for (i = 0; i < ARRAY_SIZE(ctl_idx) && i < num_ctls; i++) {
482 		global_state->ctl_to_enc_id[ctl_idx[i]] = enc_id;
483 		trace_dpu_rm_reserve_ctls(i + CTL_0, enc_id);
484 	}
485 
486 	return 0;
487 }
488 
489 static int _dpu_rm_reserve_dsc(struct dpu_rm *rm,
490 			       struct dpu_global_state *global_state,
491 			       struct drm_encoder *enc,
492 			       const struct msm_display_topology *top)
493 {
494 	int num_dsc = top->num_dsc;
495 	int i;
496 
497 	/* check if DSC required are allocated or not */
498 	for (i = 0; i < num_dsc; i++) {
499 		if (global_state->dsc_to_enc_id[i]) {
500 			DPU_ERROR("DSC %d is already allocated\n", i);
501 			return -EIO;
502 		}
503 	}
504 
505 	for (i = 0; i < num_dsc; i++)
506 		global_state->dsc_to_enc_id[i] = enc->base.id;
507 
508 	return 0;
509 }
510 
511 static int _dpu_rm_make_reservation(
512 		struct dpu_rm *rm,
513 		struct dpu_global_state *global_state,
514 		struct drm_encoder *enc,
515 		struct dpu_rm_requirements *reqs)
516 {
517 	int ret;
518 
519 	ret = _dpu_rm_reserve_lms(rm, global_state, enc->base.id, reqs);
520 	if (ret) {
521 		DPU_ERROR("unable to find appropriate mixers\n");
522 		return ret;
523 	}
524 
525 	ret = _dpu_rm_reserve_ctls(rm, global_state, enc->base.id,
526 				&reqs->topology);
527 	if (ret) {
528 		DPU_ERROR("unable to find appropriate CTL\n");
529 		return ret;
530 	}
531 
532 	ret  = _dpu_rm_reserve_dsc(rm, global_state, enc, &reqs->topology);
533 	if (ret)
534 		return ret;
535 
536 	return ret;
537 }
538 
539 static int _dpu_rm_populate_requirements(
540 		struct drm_encoder *enc,
541 		struct dpu_rm_requirements *reqs,
542 		struct msm_display_topology req_topology)
543 {
544 	reqs->topology = req_topology;
545 
546 	DRM_DEBUG_KMS("num_lm: %d num_enc: %d num_intf: %d\n",
547 		      reqs->topology.num_lm, reqs->topology.num_enc,
548 		      reqs->topology.num_intf);
549 
550 	return 0;
551 }
552 
553 static void _dpu_rm_clear_mapping(uint32_t *res_mapping, int cnt,
554 				  uint32_t enc_id)
555 {
556 	int i;
557 
558 	for (i = 0; i < cnt; i++) {
559 		if (res_mapping[i] == enc_id)
560 			res_mapping[i] = 0;
561 	}
562 }
563 
564 void dpu_rm_release(struct dpu_global_state *global_state,
565 		    struct drm_encoder *enc)
566 {
567 	_dpu_rm_clear_mapping(global_state->pingpong_to_enc_id,
568 		ARRAY_SIZE(global_state->pingpong_to_enc_id), enc->base.id);
569 	_dpu_rm_clear_mapping(global_state->mixer_to_enc_id,
570 		ARRAY_SIZE(global_state->mixer_to_enc_id), enc->base.id);
571 	_dpu_rm_clear_mapping(global_state->ctl_to_enc_id,
572 		ARRAY_SIZE(global_state->ctl_to_enc_id), enc->base.id);
573 	_dpu_rm_clear_mapping(global_state->dsc_to_enc_id,
574 		ARRAY_SIZE(global_state->dsc_to_enc_id), enc->base.id);
575 }
576 
577 int dpu_rm_reserve(
578 		struct dpu_rm *rm,
579 		struct dpu_global_state *global_state,
580 		struct drm_encoder *enc,
581 		struct drm_crtc_state *crtc_state,
582 		struct msm_display_topology topology)
583 {
584 	struct dpu_rm_requirements reqs;
585 	int ret;
586 
587 	/* Check if this is just a page-flip */
588 	if (!drm_atomic_crtc_needs_modeset(crtc_state))
589 		return 0;
590 
591 	if (IS_ERR(global_state)) {
592 		DPU_ERROR("failed to global state\n");
593 		return PTR_ERR(global_state);
594 	}
595 
596 	DRM_DEBUG_KMS("reserving hw for enc %d crtc %d\n",
597 		      enc->base.id, crtc_state->crtc->base.id);
598 
599 	ret = _dpu_rm_populate_requirements(enc, &reqs, topology);
600 	if (ret) {
601 		DPU_ERROR("failed to populate hw requirements\n");
602 		return ret;
603 	}
604 
605 	ret = _dpu_rm_make_reservation(rm, global_state, enc, &reqs);
606 	if (ret)
607 		DPU_ERROR("failed to reserve hw resources: %d\n", ret);
608 
609 
610 
611 	return ret;
612 }
613 
614 int dpu_rm_get_assigned_resources(struct dpu_rm *rm,
615 	struct dpu_global_state *global_state, uint32_t enc_id,
616 	enum dpu_hw_blk_type type, struct dpu_hw_blk **blks, int blks_size)
617 {
618 	struct dpu_hw_blk **hw_blks;
619 	uint32_t *hw_to_enc_id;
620 	int i, num_blks, max_blks;
621 
622 	switch (type) {
623 	case DPU_HW_BLK_PINGPONG:
624 		hw_blks = rm->pingpong_blks;
625 		hw_to_enc_id = global_state->pingpong_to_enc_id;
626 		max_blks = ARRAY_SIZE(rm->pingpong_blks);
627 		break;
628 	case DPU_HW_BLK_LM:
629 		hw_blks = rm->mixer_blks;
630 		hw_to_enc_id = global_state->mixer_to_enc_id;
631 		max_blks = ARRAY_SIZE(rm->mixer_blks);
632 		break;
633 	case DPU_HW_BLK_CTL:
634 		hw_blks = rm->ctl_blks;
635 		hw_to_enc_id = global_state->ctl_to_enc_id;
636 		max_blks = ARRAY_SIZE(rm->ctl_blks);
637 		break;
638 	case DPU_HW_BLK_DSPP:
639 		hw_blks = rm->dspp_blks;
640 		hw_to_enc_id = global_state->dspp_to_enc_id;
641 		max_blks = ARRAY_SIZE(rm->dspp_blks);
642 		break;
643 	case DPU_HW_BLK_DSC:
644 		hw_blks = rm->dsc_blks;
645 		hw_to_enc_id = global_state->dsc_to_enc_id;
646 		max_blks = ARRAY_SIZE(rm->dsc_blks);
647 		break;
648 	default:
649 		DPU_ERROR("blk type %d not managed by rm\n", type);
650 		return 0;
651 	}
652 
653 	num_blks = 0;
654 	for (i = 0; i < max_blks; i++) {
655 		if (hw_to_enc_id[i] != enc_id)
656 			continue;
657 
658 		if (num_blks == blks_size) {
659 			DPU_ERROR("More than %d resources assigned to enc %d\n",
660 				  blks_size, enc_id);
661 			break;
662 		}
663 		blks[num_blks++] = hw_blks[i];
664 	}
665 
666 	return num_blks;
667 }
668