6db4831e98
Android 14
198 lines
5.1 KiB
C
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);
|
|
}
|
|
}
|