kernel_samsung_a34x-permissive/drivers/gpu/drm/mediatek/mtk_writeback.c
2024-04-28 15:51:13 +02:00

198 lines
5.1 KiB
C

/* SPDX-License-Identifier: GPL-2.0 */
/*
* Copyright (c) 2019 MediaTek Inc.
*/
#include <drm/drmP.h>
#include <drm/drm_atomic_helper.h>
#include <drm/drm_crtc_helper.h>
#include <drm/drm_writeback.h>
#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);
}
}