/* SPDX-License-Identifier: GPL-2.0 */ /* * Copyright (c) 2019 MediaTek Inc. */ #include #include #include #include #include "mtk_drm_ddp.h" #include "mtk_drm_ddp_comp.h" #include "mtk_drm_drv.h" #include "mtk_drm_gem.h" #include "mtk_drm_fb.h" #include "mtk_writeback.h" #define DISP_REG_WDMA_DST_ADDR 0x0f00 #define WDMA_BUFFER_SIZE(v, h) ((v) * (h)*3) static enum mtk_ddp_comp_id mtk_wb_main[] = { DDP_COMPONENT_OD, DDP_COMPONENT_WDMA0, }; static enum mtk_ddp_comp_id mtk_wb_ext[] = { DDP_COMPONENT_OD1, DDP_COMPONENT_WDMA1, }; static const u32 wb_output_formats[] = { DRM_FORMAT_RGB888, }; static enum drm_connector_status mtk_wb_connector_detect(struct drm_connector *connector, bool force) { return connector_status_connected; } static int mtk_wb_connector_get_modes(struct drm_connector *connector) { struct drm_device *dev = connector->dev; return drm_add_modes_noedid(connector, dev->mode_config.max_width, dev->mode_config.max_height); } static enum drm_mode_status mtk_wb_connector_mode_valid(struct drm_connector *connector, struct drm_display_mode *mode) { struct drm_device *dev = connector->dev; struct drm_mode_config *mode_config = &dev->mode_config; int w = mode->hdisplay, h = mode->vdisplay; if ((w < mode_config->min_width) || (w > mode_config->max_width)) return MODE_BAD_HVALUE; if ((h < mode_config->min_height) || (h > mode_config->max_height)) return MODE_BAD_VVALUE; return MODE_OK; } static int mtk_wb_atomic_check(struct drm_encoder *encoder, struct drm_crtc_state *crtc_state, struct drm_connector_state *conn_state) { return 0; } static const struct drm_encoder_helper_funcs mtk_wb_encoder_helper_funcs = { .atomic_check = mtk_wb_atomic_check, }; static const struct drm_connector_funcs mtk_wb_connector_funcs = { /* .dpms = drm_atomic_helper_connector_dpms, */ .detect = mtk_wb_connector_detect, .fill_modes = drm_helper_probe_single_connector_modes, .destroy = drm_connector_cleanup, .reset = drm_atomic_helper_connector_reset, .atomic_duplicate_state = drm_atomic_helper_connector_duplicate_state, .atomic_destroy_state = drm_atomic_helper_connector_destroy_state, }; static const struct drm_connector_helper_funcs mtk_wb_connector_helper_funcs = { .get_modes = mtk_wb_connector_get_modes, .mode_valid = mtk_wb_connector_mode_valid, }; static struct mtk_ddp_comp *mtk_wb_find_wdma(struct mtk_drm_crtc *mtk_crtc) { struct mtk_ddp_comp *wdma = NULL; wdma = mtk_ddp_comp_find_by_id(&mtk_crtc->base, mtk_wb_main[1]); if (wdma == NULL) wdma = mtk_ddp_comp_find_by_id(&mtk_crtc->base, mtk_wb_ext[1]); return wdma; } void mtk_wb_remove_comp_from_path(struct mtk_drm_crtc *mtk_crtc) { struct mtk_ddp_comp *wdma = NULL; wdma = mtk_wb_find_wdma(mtk_crtc); if (wdma == NULL) return; drm_framebuffer_remove(wdma->fb); if (wdma->id == mtk_wb_main[1]) { mtk_ddp_remove_comp_from_path(mtk_crtc->config_regs, mtk_crtc->mmsys_reg_data, mtk_wb_main[0], mtk_wb_main[1]); } else if (wdma->id == mtk_wb_ext[1]) { mtk_ddp_remove_comp_from_path(mtk_crtc->config_regs, mtk_crtc->mmsys_reg_data, mtk_wb_ext[0], mtk_wb_ext[1]); } else { return; } } int mtk_wb_set_possible_crtcs(struct drm_device *drm_dev, struct mtk_drm_crtc *mtk_crtc, unsigned long possible_crtc) { struct drm_writeback_connector *wb_connector = &mtk_crtc->wb_connector; unsigned int *possible_crtcs = &wb_connector->encoder.possible_crtcs; *possible_crtcs = possible_crtc; if (*possible_crtcs == 0) { DRM_INFO("Failed to set wb possible_crtcs\n"); return -1; } return 0; } int mtk_wb_connector_init(struct drm_device *drm_dev, struct mtk_drm_crtc *mtk_crtc) { int ret; struct drm_writeback_connector *wb_connector = &mtk_crtc->wb_connector; ret = drm_writeback_connector_init( drm_dev, wb_connector, &mtk_wb_connector_funcs, &mtk_wb_encoder_helper_funcs, wb_output_formats, (int)ARRAY_SIZE(wb_output_formats)); if (ret != 0) return ret; drm_connector_helper_add(&mtk_crtc->wb_connector.base, &mtk_wb_connector_helper_funcs); return 0; } #ifdef CONFIG_MTK_DISPLAY_CMDQ void mtk_wb_atomic_commit(struct mtk_drm_crtc *mtk_crtc, unsigned int v, unsigned int h, void *cmdq_handle) #else void mtk_wb_atomic_commit(struct mtk_drm_crtc *mtk_crtc) #endif { struct mtk_ddp_comp *wdma = NULL; struct drm_writeback_connector *wb_conn = &mtk_crtc->wb_connector; struct drm_connector_state *conn_state = wb_conn->base.state; if (conn_state == NULL) return; if (conn_state->writeback_job != NULL && conn_state->writeback_job->fb != NULL) { struct drm_framebuffer *fb = conn_state->writeback_job->fb; u32 addr; #ifdef CONFIG_MTK_DISPLAY_CMDQ #else void *cmdq_handle = NULL; #endif wdma = mtk_wb_find_wdma(mtk_crtc); if (!wdma) return; mtk_crtc->wb_enable = true; drm_writeback_queue_job(wb_conn, conn_state->writeback_job); conn_state->writeback_job = NULL; addr = (u32)mtk_fb_get_dma(fb); if (!addr) return; mtk_ddp_write(wdma, addr & 0xFFFFFFFFU, DISP_REG_WDMA_DST_ADDR, cmdq_handle); } }