blob: f11459bff2ee2d8b2fe7300072cccc76c92aa963 [file] [log] [blame]
/* Copyright (c) 2015-2018, The Linux Foundation. All rights reserved.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2 and
* only version 2 as published by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
*/
#define pr_fmt(fmt) "%s: " fmt, __func__
#include <linux/uaccess.h>
#include <linux/delay.h>
#include <linux/file.h>
#include <linux/uaccess.h>
#include "mdss_mdp_wfd.h"
/*
* time out value for wfd to wait for any pending frames to finish
* assuming 30fps, and max 5 frames in the queue
*/
#define WFD_TIMEOUT_IN_MS 150
struct mdss_mdp_wfd *mdss_mdp_wfd_init(struct device *device,
struct mdss_mdp_ctl *ctl)
{
struct mdss_mdp_wfd *wfd;
wfd = kzalloc(sizeof(struct mdss_mdp_wfd), GFP_KERNEL);
if (!wfd)
return ERR_PTR(-ENOMEM);
mutex_init(&wfd->lock);
INIT_LIST_HEAD(&wfd->data_queue);
init_completion(&wfd->comp);
wfd->ctl = ctl;
wfd->device = device;
return wfd;
}
void mdss_mdp_wfd_deinit(struct mdss_mdp_wfd *wfd)
{
struct mdss_mdp_wfd_data *node, *temp;
list_for_each_entry_safe(node, temp, &wfd->data_queue, next)
mdss_mdp_wfd_remove_data(wfd, node);
kfree(wfd);
}
int mdss_mdp_wfd_wait_for_finish(struct mdss_mdp_wfd *wfd)
{
int ret;
mutex_lock(&wfd->lock);
if (list_empty(&wfd->data_queue)) {
mutex_unlock(&wfd->lock);
return 0;
}
init_completion(&wfd->comp);
mutex_unlock(&wfd->lock);
ret = wait_for_completion_timeout(&wfd->comp,
msecs_to_jiffies(WFD_TIMEOUT_IN_MS));
if (ret == 0)
ret = -ETIME;
else if (ret > 0)
ret = 0;
return ret;
}
void mdss_mdp_wfd_destroy(struct mdss_mdp_wfd *wfd)
{
struct mdss_mdp_ctl *ctl = wfd->ctl;
if (!ctl)
return;
if (ctl->ops.stop_fnc)
ctl->ops.stop_fnc(ctl, 0);
mdss_mdp_reset_mixercfg(ctl);
if (ctl->wb)
mdss_mdp_wb_free(ctl->wb);
if (ctl->mixer_left)
mdss_mdp_mixer_free(ctl->mixer_left);
if (ctl->mixer_right)
mdss_mdp_mixer_free(ctl->mixer_right);
ctl->mixer_left = NULL;
ctl->mixer_right = NULL;
ctl->wb = NULL;
}
bool mdss_mdp_wfd_is_config_same(struct msm_fb_data_type *mfd,
struct mdp_output_layer *layer)
{
struct mdss_overlay_private *mdp5_data = mfd_to_mdp5_data(mfd);
struct mdss_mdp_ctl *ctl = mdp5_data->wfd->ctl;
struct mdss_mdp_writeback *wb = NULL;
wb = ctl->wb;
if (!wb || !ctl->mixer_left)
return false;
if ((wb->num != layer->writeback_ndx)
|| (ctl->width != layer->buffer.width)
|| (ctl->height != layer->buffer.height)
|| (ctl->dst_format != layer->buffer.format))
return false;
return true;
}
int mdss_mdp_wfd_setup(struct mdss_mdp_wfd *wfd,
struct mdp_output_layer *layer)
{
u32 wb_idx = layer->writeback_ndx;
struct mdss_mdp_ctl *ctl = wfd->ctl;
struct mdss_mdp_writeback *wb = NULL;
struct mdss_mdp_format_params *fmt = NULL;
int ret = 0;
u32 width, height, max_mixer_width;
if (!ctl)
return -EINVAL;
if (mdss_mdp_wfd_is_config_same(ctl->mfd, layer)) {
pr_debug("wfd prepared already\n");
return 0;
}
if (ctl->wb) {
pr_debug("config change, wait for pending buffer done\n");
ret = mdss_mdp_wfd_wait_for_finish(wfd);
if (ret) {
pr_err("fail to wait for outstanding request\n");
return ret;
}
mdss_mdp_wfd_destroy(wfd);
}
width = layer->buffer.width;
height = layer->buffer.height;
max_mixer_width = ctl->mdata->max_mixer_width;
pr_debug("widthxheight:%dx%d,wb_idx:%d, ctl:%d\n", width, height,
wb_idx, ctl->num);
wb = mdss_mdp_wb_assign(wb_idx, ctl->num);
if (!wb) {
pr_err("could not allocate wb\n");
ret = -EINVAL;
goto wfd_setup_error;
}
ctl->wb = wb;
ctl->dst_format = layer->buffer.format;
ctl->dst_comp_ratio = layer->buffer.comp_ratio;
ctl->width = width;
ctl->height = height;
ctl->roi = (struct mdss_rect) {0, 0, width, height};
ctl->is_secure = (layer->flags & MDP_LAYER_SECURE_SESSION);
fmt = mdss_mdp_get_format_params(layer->buffer.format);
if (fmt == NULL) {
pr_err("invalid buffer format\n");
ret = -EINVAL;
goto wfd_setup_error;
}
/* only 3 csc type supported */
if (fmt->is_yuv) {
switch (layer->color_space) {
case MDP_CSC_ITU_R_601:
ctl->csc_type = MDSS_MDP_CSC_RGB2YUV_601L;
break;
case MDP_CSC_ITU_R_709:
ctl->csc_type = MDSS_MDP_CSC_RGB2YUV_709L;
break;
case MDP_CSC_ITU_R_601_FR:
default:
ctl->csc_type = MDSS_MDP_CSC_RGB2YUV_601FR;
break;
}
} else {
ctl->csc_type = MDSS_MDP_CSC_RGB2RGB;
}
if (ctl->mdata->wfd_mode == MDSS_MDP_WFD_INTERFACE) {
ctl->mixer_left = mdss_mdp_mixer_alloc(ctl,
MDSS_MDP_MIXER_TYPE_INTF, (width > max_mixer_width), 0);
if (width > max_mixer_width) {
ctl->mixer_right = mdss_mdp_mixer_alloc(ctl,
MDSS_MDP_MIXER_TYPE_INTF, true, 0);
ctl->mfd->split_mode = MDP_DUAL_LM_SINGLE_DISPLAY;
width = width / 2;
} else {
ctl->mfd->split_mode = MDP_SPLIT_MODE_NONE;
}
} else if (width > max_mixer_width) {
pr_err("width > max_mixer_width supported only in MDSS_MDP_WB_INTF\n");
goto wfd_setup_error;
} else if (ctl->mdata->wfd_mode == MDSS_MDP_WFD_DEDICATED) {
ctl->mixer_left = mdss_mdp_mixer_alloc(ctl,
MDSS_MDP_MIXER_TYPE_WRITEBACK, false, 0);
} else {
ctl->mixer_left = mdss_mdp_mixer_assign(wb->num, true, false);
}
if (!ctl->mixer_left ||
((ctl->mfd->split_mode ==
MDP_DUAL_LM_SINGLE_DISPLAY) && (!ctl->mixer_right))) {
if (ctl->mixer_left)
mdss_mdp_mixer_free(ctl->mixer_left);
if (ctl->mixer_right)
mdss_mdp_mixer_free(ctl->mixer_right);
pr_err("could not allocate mixer(s) for ctl:%d\n", ctl->num);
ret = -ENODEV;
goto wfd_setup_error;
}
if (ctl->mixer_left->type == MDSS_MDP_MIXER_TYPE_INTF ||
ctl->mdata->wfd_mode == MDSS_MDP_WFD_DEDICATED) {
ctl->opmode = MDSS_MDP_CTL_OP_WFD_MODE;
} else {
switch (ctl->mixer_left->num) {
case MDSS_MDP_WB_LAYERMIXER0:
ctl->opmode = MDSS_MDP_CTL_OP_WB0_MODE;
break;
case MDSS_MDP_WB_LAYERMIXER1:
ctl->opmode = MDSS_MDP_CTL_OP_WB1_MODE;
break;
default:
pr_err("Incorrect writeback config num=%d\n",
ctl->mixer_left->num);
ret = -EINVAL;
goto wfd_setup_error;
}
ctl->wb_type = MDSS_MDP_WB_CTL_TYPE_LINE;
}
ctl->mixer_left->width = width;
ctl->mixer_left->height = height;
ctl->mixer_left->roi = (struct mdss_rect) {0, 0, width, height};
ctl->mixer_left->ctl = ctl;
ctl->mixer_left->valid_roi = true;
ctl->mixer_left->roi_changed = true;
if (ctl->mfd->split_mode == MDP_DUAL_LM_SINGLE_DISPLAY) {
ctl->mixer_right->width = width;
ctl->mixer_right->height = height;
ctl->mixer_right->roi = (struct mdss_rect) {0, 0,
width, height};
ctl->mixer_right->valid_roi = true;
ctl->mixer_right->roi_changed = true;
ctl->mixer_right->ctl = ctl;
ctl->opmode |= MDSS_MDP_CTL_OP_PACK_3D_ENABLE |
MDSS_MDP_CTL_OP_PACK_3D_H_ROW_INT;
} else {
ctl->opmode &= ~(MDSS_MDP_CTL_OP_PACK_3D_ENABLE |
MDSS_MDP_CTL_OP_PACK_3D_H_ROW_INT);
}
if (ctl->ops.start_fnc) {
ret = ctl->ops.start_fnc(ctl);
if (ret) {
pr_err("wfd start failed %d\n", ret);
goto wfd_setup_error;
}
}
return ret;
wfd_setup_error:
mdss_mdp_wfd_destroy(wfd);
return ret;
}
static int mdss_mdp_wfd_import_data(struct device *device,
struct mdss_mdp_wfd_data *wfd_data)
{
int i, ret = 0;
u32 flags = 0;
struct mdp_layer_buffer *buffer = &wfd_data->layer.buffer;
struct mdss_mdp_data *data = &wfd_data->data;
struct msmfb_data planes[MAX_PLANES];
if (wfd_data->layer.flags & MDP_LAYER_SECURE_SESSION)
flags = MDP_SECURE_OVERLAY_SESSION;
if (buffer->plane_count > MAX_PLANES) {
pr_err("buffer plane_count exceeds MAX_PLANES limit:%d",
buffer->plane_count);
return -EINVAL;
}
memset(planes, 0, sizeof(planes));
for (i = 0; i < buffer->plane_count; i++) {
planes[i].memory_id = buffer->planes[i].fd;
planes[i].offset = buffer->planes[i].offset;
}
ret = mdss_mdp_data_get_and_validate_size(data, planes,
buffer->plane_count, flags, device,
false, DMA_FROM_DEVICE, buffer);
return ret;
}
struct mdss_mdp_wfd_data *mdss_mdp_wfd_add_data(
struct mdss_mdp_wfd *wfd,
struct mdp_output_layer *layer)
{
int ret;
struct mdss_mdp_wfd_data *wfd_data;
if (!wfd->ctl || !wfd->ctl->wb) {
pr_err("wfd not setup\n");
return ERR_PTR(-EINVAL);
}
wfd_data = kzalloc(sizeof(struct mdss_mdp_wfd_data), GFP_KERNEL);
if (!wfd_data)
return ERR_PTR(-ENOMEM);
wfd_data->layer = *layer;
ret = mdss_mdp_wfd_import_data(wfd->device, wfd_data);
if (ret) {
pr_err("fail to import data\n");
mdss_mdp_data_free(&wfd_data->data, true, DMA_FROM_DEVICE);
kfree(wfd_data);
return ERR_PTR(ret);
}
mutex_lock(&wfd->lock);
list_add_tail(&wfd_data->next, &wfd->data_queue);
mutex_unlock(&wfd->lock);
return wfd_data;
}
void mdss_mdp_wfd_remove_data(struct mdss_mdp_wfd *wfd,
struct mdss_mdp_wfd_data *wfd_data)
{
mutex_lock(&wfd->lock);
list_del_init(&wfd_data->next);
if (list_empty(&wfd->data_queue))
complete(&wfd->comp);
mutex_unlock(&wfd->lock);
mdss_mdp_data_free(&wfd_data->data, true, DMA_FROM_DEVICE);
kfree(wfd_data);
}
static int mdss_mdp_wfd_validate_out_configuration(struct mdss_mdp_wfd *wfd,
struct mdp_output_layer *layer)
{
struct mdss_mdp_format_params *fmt = NULL;
struct mdss_mdp_ctl *ctl = wfd->ctl;
u32 wb_idx = layer->writeback_ndx;
if (mdss_mdp_is_wb_mdp_intf(wb_idx, ctl->num)) {
fmt = mdss_mdp_get_format_params(layer->buffer.format);
if (fmt && !(fmt->flag & VALID_MDP_WB_INTF_FORMAT)) {
pr_err("wb=%d does not support dst fmt:%d\n", wb_idx,
layer->buffer.format);
return -EINVAL;
}
if (!ctl->mdata->has_wb_ubwc && mdss_mdp_is_ubwc_format(fmt)) {
pr_err("wb=%d does not support UBWC fmt:%d\n", wb_idx,
layer->buffer.format);
return -EINVAL;
}
}
return 0;
}
int mdss_mdp_wfd_validate(struct mdss_mdp_wfd *wfd,
struct mdp_output_layer *layer)
{
u32 wb_idx = layer->writeback_ndx;
if (mdss_mdp_wfd_validate_out_configuration(wfd, layer)) {
pr_err("failed to validate output config\n");
return -EINVAL;
}
if (wb_idx > wfd->ctl->mdata->nwb)
return -EINVAL;
return 0;
}
int mdss_mdp_wfd_kickoff(struct mdss_mdp_wfd *wfd,
struct mdss_mdp_commit_cb *commit_cb)
{
struct mdss_mdp_ctl *ctl = wfd->ctl;
struct mdss_mdp_writeback_arg wb_args;
struct mdss_mdp_wfd_data *wfd_data;
int ret = 0;
if (!ctl) {
pr_err("no ctl\n");
return -EINVAL;
}
if (!ctl->wb) {
pr_err("wfd not prepared\n");
return -EINVAL;
}
mutex_lock(&wfd->lock);
if (list_empty(&wfd->data_queue)) {
pr_debug("no output buffer\n");
mutex_unlock(&wfd->lock);
mdss_mdp_ctl_notify(ctl, MDP_NOTIFY_FRAME_DONE);
return 0;
}
wfd_data = list_first_entry(&wfd->data_queue,
struct mdss_mdp_wfd_data, next);
mutex_unlock(&wfd->lock);
ret = mdss_mdp_data_map(&wfd_data->data, true, DMA_FROM_DEVICE);
if (ret) {
pr_err("fail to acquire output buffer\n");
goto kickoff_error;
}
memset(&wb_args, 0, sizeof(wb_args));
wb_args.data = &wfd_data->data;
ret = mdss_mdp_writeback_display_commit(ctl, &wb_args);
if (ret) {
pr_err("wfd commit error = %d, ctl=%d\n", ret, ctl->num);
goto kickoff_error;
}
if (commit_cb)
commit_cb->commit_cb_fnc(
MDP_COMMIT_STAGE_SETUP_DONE,
commit_cb->data);
ret = mdss_mdp_display_wait4comp(ctl);
if (commit_cb)
commit_cb->commit_cb_fnc(MDP_COMMIT_STAGE_READY_FOR_KICKOFF,
commit_cb->data);
kickoff_error:
mdss_mdp_wfd_commit_done(wfd);
return ret;
}
int mdss_mdp_wfd_commit_done(struct mdss_mdp_wfd *wfd)
{
struct mdss_mdp_wfd_data *wfd_data;
mutex_lock(&wfd->lock);
if (list_empty(&wfd->data_queue)) {
pr_err("no output buffer\n");
mutex_unlock(&wfd->lock);
return -EINVAL;
}
wfd_data = list_first_entry(&wfd->data_queue,
struct mdss_mdp_wfd_data, next);
mutex_unlock(&wfd->lock);
mdss_mdp_wfd_remove_data(wfd, wfd_data);
return 0;
}