| /* Copyright (c) 2012-2013, 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/errno.h> |
| #include <linux/kernel.h> |
| #include <linux/major.h> |
| #include <linux/module.h> |
| #include <linux/uaccess.h> |
| #include <linux/iommu.h> |
| |
| #include <mach/iommu.h> |
| #include <mach/iommu_domains.h> |
| |
| #include "mdss_mdp.h" |
| #include "mdss_fb.h" |
| #include "mdss_wb.h" |
| |
| |
| enum mdss_mdp_wb_state { |
| WB_OPEN, |
| WB_START, |
| WB_STOPING, |
| WB_STOP |
| }; |
| |
| struct mdss_mdp_wb { |
| u32 fb_ndx; |
| struct mutex lock; |
| struct list_head busy_queue; |
| struct list_head free_queue; |
| struct list_head register_queue; |
| wait_queue_head_t wait_q; |
| u32 state; |
| int is_secure; |
| struct mdss_mdp_pipe *secure_pipe; |
| }; |
| |
| enum mdss_mdp_wb_node_state { |
| REGISTERED, |
| IN_FREE_QUEUE, |
| IN_BUSY_QUEUE, |
| WITH_CLIENT |
| }; |
| |
| struct mdss_mdp_wb_data { |
| struct list_head registered_entry; |
| struct list_head active_entry; |
| struct msmfb_data buf_info; |
| struct mdss_mdp_data buf_data; |
| int state; |
| }; |
| |
| static DEFINE_MUTEX(mdss_mdp_wb_buf_lock); |
| static struct mdss_mdp_wb mdss_mdp_wb_info; |
| |
| #ifdef DEBUG_WRITEBACK |
| /* for debugging: writeback output buffer to allocated memory */ |
| static inline |
| struct mdss_mdp_data *mdss_mdp_wb_debug_buffer(struct msm_fb_data_type *mfd) |
| { |
| static struct ion_handle *ihdl; |
| static void *videomemory; |
| static ion_phys_addr_t mdss_wb_mem; |
| static struct mdss_mdp_data mdss_wb_buffer = { .num_planes = 1, }; |
| int rc; |
| |
| if (IS_ERR_OR_NULL(ihdl)) { |
| struct fb_info *fbi; |
| size_t img_size; |
| struct ion_client *iclient = mdss_get_ionclient(); |
| struct mdss_mdp_img_data *img = mdss_wb_buffer.p; |
| |
| fbi = mfd->fbi; |
| img_size = fbi->var.xres * fbi->var.yres * |
| fbi->var.bits_per_pixel / 8; |
| |
| |
| ihdl = ion_alloc(iclient, img_size, SZ_4K, |
| ION_HEAP(ION_SF_HEAP_ID), 0); |
| if (IS_ERR_OR_NULL(ihdl)) { |
| pr_err("unable to alloc fbmem from ion (%p)\n", ihdl); |
| return NULL; |
| } |
| |
| videomemory = ion_map_kernel(iclient, ihdl); |
| ion_phys(iclient, ihdl, &mdss_wb_mem, &img_size); |
| |
| if (is_mdss_iommu_attached()) { |
| int domain = MDSS_IOMMU_DOMAIN_UNSECURE; |
| rc = ion_map_iommu(iclient, ihdl, |
| mdss_get_iommu_domain(domain), |
| 0, SZ_4K, 0, |
| (unsigned long *) &img->addr, |
| (unsigned long *) &img->len, |
| 0, 0); |
| } else { |
| img->addr = mdss_wb_mem; |
| img->len = img_size; |
| } |
| |
| pr_debug("ihdl=%p virt=%p phys=0x%lx iova=0x%x size=%u\n", |
| ihdl, videomemory, mdss_wb_mem, img->addr, img_size); |
| } |
| return &mdss_wb_buffer; |
| } |
| #else |
| static inline |
| struct mdss_mdp_data *mdss_mdp_wb_debug_buffer(struct msm_fb_data_type *mfd) |
| { |
| return NULL; |
| } |
| #endif |
| |
| int mdss_mdp_wb_set_secure(struct msm_fb_data_type *mfd, int enable) |
| { |
| struct mdss_mdp_wb *wb = mfd_to_wb(mfd); |
| struct mdss_mdp_ctl *ctl = mfd_to_ctl(mfd); |
| struct mdss_mdp_pipe *pipe; |
| struct mdss_mdp_mixer *mixer; |
| |
| pr_debug("setting secure=%d\n", enable); |
| |
| ctl->is_secure = enable; |
| wb->is_secure = enable; |
| |
| /* newer revisions don't require secure src pipe for secure session */ |
| if (ctl->mdata->mdp_rev > MDSS_MDP_HW_REV_100) |
| return 0; |
| |
| pipe = wb->secure_pipe; |
| |
| if (!enable) { |
| if (pipe) { |
| /* unset pipe */ |
| mdss_mdp_mixer_pipe_unstage(pipe); |
| mdss_mdp_pipe_destroy(pipe); |
| wb->secure_pipe = NULL; |
| } |
| return 0; |
| } |
| |
| mixer = mdss_mdp_mixer_get(ctl, MDSS_MDP_MIXER_MUX_DEFAULT); |
| if (!mixer) { |
| pr_err("Unable to find mixer for wb\n"); |
| return -ENOENT; |
| } |
| |
| if (!pipe) { |
| pipe = mdss_mdp_pipe_alloc(mixer, MDSS_MDP_PIPE_TYPE_RGB); |
| if (!pipe) |
| pipe = mdss_mdp_pipe_alloc(mixer, |
| MDSS_MDP_PIPE_TYPE_VIG); |
| if (!pipe) { |
| pr_err("Unable to get pipe to set secure session\n"); |
| return -ENOMEM; |
| } |
| |
| pipe->src_fmt = mdss_mdp_get_format_params(MDP_RGBA_8888); |
| |
| pipe->mfd = mfd; |
| pipe->mixer_stage = MDSS_MDP_STAGE_BASE; |
| wb->secure_pipe = pipe; |
| } |
| |
| pipe->img_height = mixer->height; |
| pipe->img_width = mixer->width; |
| pipe->src.x = 0; |
| pipe->src.y = 0; |
| pipe->src.w = pipe->img_width; |
| pipe->src.h = pipe->img_height; |
| pipe->dst = pipe->src; |
| |
| pipe->flags = (enable ? MDP_SECURE_OVERLAY_SESSION : 0); |
| pipe->params_changed++; |
| |
| pr_debug("setting secure pipe=%d flags=%x\n", pipe->num, pipe->flags); |
| |
| return mdss_mdp_pipe_queue_data(pipe, NULL); |
| } |
| |
| static int mdss_mdp_wb_init(struct msm_fb_data_type *mfd) |
| { |
| struct mdss_overlay_private *mdp5_data = mfd_to_mdp5_data(mfd); |
| struct mdss_mdp_wb *wb = mfd_to_wb(mfd); |
| int rc = 0; |
| |
| mutex_lock(&mdss_mdp_wb_buf_lock); |
| if (wb == NULL) { |
| wb = &mdss_mdp_wb_info; |
| wb->fb_ndx = mfd->index; |
| mdp5_data->wb = wb; |
| } else if (mfd->index != wb->fb_ndx) { |
| pr_err("only one writeback intf supported at a time\n"); |
| rc = -EMLINK; |
| goto error; |
| } else { |
| pr_debug("writeback already initialized\n"); |
| } |
| |
| pr_debug("init writeback on fb%d\n", wb->fb_ndx); |
| |
| mutex_init(&wb->lock); |
| INIT_LIST_HEAD(&wb->free_queue); |
| INIT_LIST_HEAD(&wb->busy_queue); |
| INIT_LIST_HEAD(&wb->register_queue); |
| wb->state = WB_OPEN; |
| init_waitqueue_head(&wb->wait_q); |
| |
| mdp5_data->wb = wb; |
| error: |
| mutex_unlock(&mdss_mdp_wb_buf_lock); |
| return rc; |
| } |
| |
| static int mdss_mdp_wb_terminate(struct msm_fb_data_type *mfd) |
| { |
| struct mdss_overlay_private *mdp5_data = mfd_to_mdp5_data(mfd); |
| struct mdss_mdp_wb *wb = mfd_to_wb(mfd); |
| |
| if (!wb) { |
| pr_err("unable to terminate, writeback is not initialized\n"); |
| return -ENODEV; |
| } |
| |
| pr_debug("terminate writeback\n"); |
| |
| mutex_lock(&mdss_mdp_wb_buf_lock); |
| mutex_lock(&wb->lock); |
| if (!list_empty(&wb->register_queue)) { |
| struct mdss_mdp_wb_data *node, *temp; |
| list_for_each_entry_safe(node, temp, &wb->register_queue, |
| registered_entry) { |
| list_del(&node->registered_entry); |
| kfree(node); |
| } |
| } |
| |
| wb->is_secure = false; |
| if (wb->secure_pipe) |
| mdss_mdp_pipe_destroy(wb->secure_pipe); |
| mutex_unlock(&wb->lock); |
| if (mdp5_data->ctl) |
| mdp5_data->ctl->is_secure = false; |
| mdp5_data->wb = NULL; |
| mutex_unlock(&mdss_mdp_wb_buf_lock); |
| |
| return 0; |
| } |
| |
| static int mdss_mdp_wb_start(struct msm_fb_data_type *mfd) |
| { |
| struct mdss_mdp_wb *wb = mfd_to_wb(mfd); |
| |
| if (!wb) { |
| pr_err("unable to start, writeback is not initialized\n"); |
| return -ENODEV; |
| } |
| |
| mutex_lock(&wb->lock); |
| wb->state = WB_START; |
| mutex_unlock(&wb->lock); |
| wake_up(&wb->wait_q); |
| |
| return 0; |
| } |
| |
| static int mdss_mdp_wb_stop(struct msm_fb_data_type *mfd) |
| { |
| struct mdss_mdp_wb *wb = mfd_to_wb(mfd); |
| |
| if (!wb) { |
| pr_err("unable to stop, writeback is not initialized\n"); |
| return -ENODEV; |
| } |
| |
| mutex_lock(&wb->lock); |
| wb->state = WB_STOPING; |
| mutex_unlock(&wb->lock); |
| wake_up(&wb->wait_q); |
| |
| return 0; |
| } |
| |
| static int mdss_mdp_wb_register_node(struct mdss_mdp_wb *wb, |
| struct mdss_mdp_wb_data *node) |
| { |
| node->state = REGISTERED; |
| list_add_tail(&node->registered_entry, &wb->register_queue); |
| if (!node) { |
| pr_err("Invalid wb node\n"); |
| return -EINVAL; |
| } |
| |
| return 0; |
| } |
| |
| static struct mdss_mdp_wb_data *get_local_node(struct mdss_mdp_wb *wb, |
| struct msmfb_data *data) { |
| struct mdss_mdp_wb_data *node; |
| struct mdss_mdp_img_data *buf; |
| int ret; |
| |
| if (!data->iova) |
| return NULL; |
| |
| if (!list_empty(&wb->register_queue)) { |
| list_for_each_entry(node, &wb->register_queue, registered_entry) |
| if (node->buf_info.iova == data->iova) { |
| pr_debug("found node iova=%x addr=%x\n", |
| data->iova, node->buf_data.p[0].addr); |
| return node; |
| } |
| } |
| |
| node = kzalloc(sizeof(struct mdss_mdp_wb_data), GFP_KERNEL); |
| if (node == NULL) { |
| pr_err("out of memory\n"); |
| return NULL; |
| } |
| |
| node->buf_data.num_planes = 1; |
| node->buf_info = *data; |
| buf = &node->buf_data.p[0]; |
| buf->addr = (u32) (data->iova + data->offset); |
| buf->len = UINT_MAX; /* trusted source */ |
| if (wb->is_secure) |
| buf->flags |= MDP_SECURE_OVERLAY_SESSION; |
| ret = mdss_mdp_wb_register_node(wb, node); |
| if (IS_ERR_VALUE(ret)) { |
| pr_err("error registering wb node\n"); |
| kfree(node); |
| return NULL; |
| } |
| |
| pr_debug("register node iova=0x%x addr=0x%x\n", data->iova, buf->addr); |
| |
| return node; |
| } |
| |
| static struct mdss_mdp_wb_data *get_user_node(struct msm_fb_data_type *mfd, |
| struct msmfb_data *data) |
| { |
| |
| struct mdss_mdp_wb *wb = mfd_to_wb(mfd); |
| struct mdss_mdp_wb_data *node; |
| struct mdss_mdp_img_data *buf; |
| int ret; |
| |
| node = kzalloc(sizeof(struct mdss_mdp_wb_data), GFP_KERNEL); |
| if (node == NULL) { |
| pr_err("out of memory\n"); |
| return NULL; |
| } |
| |
| node->buf_data.num_planes = 1; |
| buf = &node->buf_data.p[0]; |
| if (wb->is_secure) |
| buf->flags |= MDP_SECURE_OVERLAY_SESSION; |
| ret = mdss_mdp_get_img(data, buf); |
| if (IS_ERR_VALUE(ret)) { |
| pr_err("error getting buffer info\n"); |
| goto register_fail; |
| } |
| memcpy(&node->buf_info, data, sizeof(*data)); |
| |
| ret = mdss_mdp_wb_register_node(wb, node); |
| if (IS_ERR_VALUE(ret)) { |
| pr_err("error registering wb node\n"); |
| goto register_fail; |
| } |
| |
| pr_debug("register node mem_id=%d offset=%u addr=0x%x len=%d\n", |
| data->memory_id, data->offset, buf->addr, buf->len); |
| |
| return node; |
| |
| register_fail: |
| kfree(node); |
| return NULL; |
| } |
| |
| static int mdss_mdp_wb_queue(struct msm_fb_data_type *mfd, |
| struct msmfb_data *data, int local) |
| { |
| struct mdss_mdp_wb *wb = mfd_to_wb(mfd); |
| struct mdss_mdp_wb_data *node = NULL; |
| int ret = 0; |
| |
| if (!wb) { |
| pr_err("unable to queue, writeback is not initialized\n"); |
| return -ENODEV; |
| } |
| |
| pr_debug("fb%d queue\n", wb->fb_ndx); |
| |
| mutex_lock(&wb->lock); |
| if (local) |
| node = get_local_node(wb, data); |
| if (node == NULL) |
| node = get_user_node(mfd, data); |
| |
| if (!node || node->state == IN_BUSY_QUEUE || |
| node->state == IN_FREE_QUEUE) { |
| pr_err("memory not registered or Buffer already with us\n"); |
| ret = -EINVAL; |
| } else { |
| list_add_tail(&node->active_entry, &wb->free_queue); |
| node->state = IN_FREE_QUEUE; |
| } |
| mutex_unlock(&wb->lock); |
| |
| return ret; |
| } |
| |
| static int is_buffer_ready(struct mdss_mdp_wb *wb) |
| { |
| int rc; |
| mutex_lock(&wb->lock); |
| rc = !list_empty(&wb->busy_queue) || (wb->state == WB_STOPING); |
| mutex_unlock(&wb->lock); |
| |
| return rc; |
| } |
| |
| static int mdss_mdp_wb_dequeue(struct msm_fb_data_type *mfd, |
| struct msmfb_data *data) |
| { |
| struct mdss_mdp_wb *wb = mfd_to_wb(mfd); |
| struct mdss_mdp_wb_data *node = NULL; |
| int ret; |
| |
| if (!wb) { |
| pr_err("unable to dequeue, writeback is not initialized\n"); |
| return -ENODEV; |
| } |
| |
| ret = wait_event_interruptible(wb->wait_q, is_buffer_ready(wb)); |
| if (ret) { |
| pr_err("failed to get dequeued buffer\n"); |
| return -ENOBUFS; |
| } |
| |
| mutex_lock(&wb->lock); |
| if (wb->state == WB_STOPING) { |
| pr_debug("wfd stopped\n"); |
| wb->state = WB_STOP; |
| ret = -ENOBUFS; |
| } else if (!list_empty(&wb->busy_queue)) { |
| struct mdss_mdp_img_data *buf; |
| node = list_first_entry(&wb->busy_queue, |
| struct mdss_mdp_wb_data, |
| active_entry); |
| list_del(&node->active_entry); |
| node->state = WITH_CLIENT; |
| memcpy(data, &node->buf_info, sizeof(*data)); |
| |
| buf = &node->buf_data.p[0]; |
| pr_debug("found node addr=%x len=%d\n", buf->addr, buf->len); |
| } else { |
| pr_debug("node is NULL, wait for next\n"); |
| ret = -ENOBUFS; |
| } |
| mutex_unlock(&wb->lock); |
| return ret; |
| } |
| |
| static void mdss_mdp_wb_callback(void *arg) |
| { |
| if (arg) |
| complete((struct completion *) arg); |
| } |
| |
| int mdss_mdp_wb_kickoff(struct msm_fb_data_type *mfd) |
| { |
| struct mdss_mdp_wb *wb = mfd_to_wb(mfd); |
| struct mdss_mdp_ctl *ctl = mfd_to_ctl(mfd); |
| struct mdss_mdp_wb_data *node = NULL; |
| int ret = 0; |
| DECLARE_COMPLETION_ONSTACK(comp); |
| struct mdss_mdp_writeback_arg wb_args = { |
| .callback_fnc = mdss_mdp_wb_callback, |
| .priv_data = &comp, |
| }; |
| |
| if (!ctl->power_on) |
| return 0; |
| |
| mutex_lock(&mdss_mdp_wb_buf_lock); |
| if (wb) { |
| mutex_lock(&wb->lock); |
| /* in case of reinit of control path need to reset secure */ |
| if (ctl->play_cnt == 0) |
| mdss_mdp_wb_set_secure(ctl->mfd, wb->is_secure); |
| if (!list_empty(&wb->free_queue) && wb->state != WB_STOPING && |
| wb->state != WB_STOP) { |
| node = list_first_entry(&wb->free_queue, |
| struct mdss_mdp_wb_data, |
| active_entry); |
| list_del(&node->active_entry); |
| node->state = IN_BUSY_QUEUE; |
| wb_args.data = &node->buf_data; |
| } else { |
| pr_debug("unable to get buf wb state=%d\n", wb->state); |
| } |
| mutex_unlock(&wb->lock); |
| } |
| |
| if (wb_args.data == NULL) |
| wb_args.data = mdss_mdp_wb_debug_buffer(ctl->mfd); |
| |
| if (wb_args.data == NULL) { |
| pr_err("unable to get writeback buf ctl=%d\n", ctl->num); |
| /* drop buffer but don't return error */ |
| ret = 0; |
| goto kickoff_fail; |
| } |
| |
| ret = mdss_mdp_writeback_display_commit(ctl, &wb_args); |
| if (ret) { |
| pr_err("error on commit ctl=%d\n", ctl->num); |
| goto kickoff_fail; |
| } |
| |
| ret = wait_for_completion_timeout(&comp, KOFF_TIMEOUT); |
| if (ret == 0) |
| WARN(1, "wfd kick off time out=%d ctl=%d", ret, ctl->num); |
| else |
| ret = 0; |
| |
| if (wb && node) { |
| mutex_lock(&wb->lock); |
| list_add_tail(&node->active_entry, &wb->busy_queue); |
| mutex_unlock(&wb->lock); |
| wake_up(&wb->wait_q); |
| } |
| |
| kickoff_fail: |
| mutex_unlock(&mdss_mdp_wb_buf_lock); |
| return ret; |
| } |
| |
| int mdss_mdp_wb_set_mirr_hint(struct msm_fb_data_type *mfd, int hint) |
| { |
| struct mdss_panel_data *pdata = NULL; |
| struct mdss_wb_ctrl *wb_ctrl = NULL; |
| |
| if (!mfd) { |
| pr_err("No panel data!\n"); |
| return -EINVAL; |
| } |
| |
| pdata = mfd->pdev->dev.platform_data; |
| wb_ctrl = container_of(pdata, struct mdss_wb_ctrl, pdata); |
| |
| switch (hint) { |
| case MDP_WRITEBACK_MIRROR_ON: |
| case MDP_WRITEBACK_MIRROR_PAUSE: |
| case MDP_WRITEBACK_MIRROR_RESUME: |
| case MDP_WRITEBACK_MIRROR_OFF: |
| pr_info("wfd state switched to %d\n", hint); |
| switch_set_state(&wb_ctrl->sdev, hint); |
| return 0; |
| default: |
| return -EINVAL; |
| } |
| } |
| |
| int mdss_mdp_wb_get_format(struct msm_fb_data_type *mfd, |
| struct mdp_mixer_cfg *mixer_cfg) |
| { |
| int dst_format; |
| struct mdss_mdp_ctl *ctl = mfd_to_ctl(mfd); |
| |
| if (!ctl) { |
| pr_err("No panel data!\n"); |
| return -EINVAL; |
| } |
| |
| switch (ctl->dst_format) { |
| case MDP_RGB_888: |
| dst_format = WB_FORMAT_RGB_888; |
| break; |
| case MDP_RGB_565: |
| dst_format = WB_FORMAT_RGB_565; |
| break; |
| case MDP_XRGB_8888: |
| dst_format = WB_FORMAT_xRGB_8888; |
| break; |
| case MDP_ARGB_8888: |
| dst_format = WB_FORMAT_ARGB_8888; |
| break; |
| case MDP_BGRA_8888: |
| dst_format = WB_FORMAT_BGRA_8888; |
| break; |
| case MDP_BGRX_8888: |
| dst_format = WB_FORMAT_BGRX_8888; |
| break; |
| case MDP_Y_CBCR_H2V2_VENUS: |
| dst_format = WB_FORMAT_NV12; |
| break; |
| default: |
| return -EINVAL; |
| } |
| mixer_cfg->writeback_format = dst_format; |
| return 0; |
| } |
| |
| int mdss_mdp_wb_set_format(struct msm_fb_data_type *mfd, int dst_format) |
| { |
| struct mdss_mdp_ctl *ctl = mfd_to_ctl(mfd); |
| |
| if (!ctl) { |
| pr_err("No panel data!\n"); |
| return -EINVAL; |
| } |
| |
| switch (dst_format) { |
| case WB_FORMAT_RGB_888: |
| ctl->dst_format = MDP_RGB_888; |
| break; |
| case WB_FORMAT_RGB_565: |
| ctl->dst_format = MDP_RGB_565; |
| break; |
| case WB_FORMAT_xRGB_8888: |
| ctl->dst_format = MDP_XRGB_8888; |
| break; |
| case WB_FORMAT_ARGB_8888: |
| ctl->dst_format = MDP_ARGB_8888; |
| break; |
| case WB_FORMAT_BGRA_8888: |
| ctl->dst_format = MDP_BGRA_8888; |
| break; |
| case WB_FORMAT_BGRX_8888: |
| ctl->dst_format = MDP_BGRX_8888; |
| break; |
| case WB_FORMAT_NV12: |
| ctl->dst_format = MDP_Y_CBCR_H2V2_VENUS; |
| break; |
| default: |
| pr_err("wfd format not supported\n"); |
| return -EINVAL; |
| } |
| |
| pr_debug("wfd format %d\n", ctl->dst_format); |
| return 0; |
| } |
| |
| int mdss_mdp_wb_ioctl_handler(struct msm_fb_data_type *mfd, u32 cmd, |
| void *arg) |
| { |
| struct msmfb_data data; |
| int ret = -ENOSYS, hint = 0; |
| |
| switch (cmd) { |
| case MSMFB_WRITEBACK_INIT: |
| ret = mdss_mdp_wb_init(mfd); |
| break; |
| case MSMFB_WRITEBACK_START: |
| ret = mdss_mdp_wb_start(mfd); |
| break; |
| case MSMFB_WRITEBACK_STOP: |
| ret = mdss_mdp_wb_stop(mfd); |
| break; |
| case MSMFB_WRITEBACK_QUEUE_BUFFER: |
| if (!copy_from_user(&data, arg, sizeof(data))) { |
| ret = mdss_mdp_wb_queue(mfd, &data, false); |
| ret = copy_to_user(arg, &data, sizeof(data)); |
| } else { |
| pr_err("wb queue buf failed on copy_from_user\n"); |
| ret = -EFAULT; |
| } |
| break; |
| case MSMFB_WRITEBACK_DEQUEUE_BUFFER: |
| if (!copy_from_user(&data, arg, sizeof(data))) { |
| ret = mdss_mdp_wb_dequeue(mfd, &data); |
| ret = copy_to_user(arg, &data, sizeof(data)); |
| } else { |
| pr_err("wb dequeue buf failed on copy_from_user\n"); |
| ret = -EFAULT; |
| } |
| break; |
| case MSMFB_WRITEBACK_TERMINATE: |
| ret = mdss_mdp_wb_terminate(mfd); |
| break; |
| case MSMFB_WRITEBACK_SET_MIRRORING_HINT: |
| if (!copy_from_user(&hint, arg, sizeof(hint))) { |
| ret = mdss_mdp_wb_set_mirr_hint(mfd, hint); |
| } else { |
| pr_err("set mirroring hint failed on copy_from_user\n"); |
| ret = -EFAULT; |
| } |
| break; |
| } |
| |
| return ret; |
| } |
| |
| int msm_fb_writeback_start(struct fb_info *info) |
| { |
| struct msm_fb_data_type *mfd = (struct msm_fb_data_type *) info->par; |
| |
| if (!mfd) |
| return -ENODEV; |
| |
| return mdss_mdp_wb_start(mfd); |
| } |
| EXPORT_SYMBOL(msm_fb_writeback_start); |
| |
| int msm_fb_writeback_queue_buffer(struct fb_info *info, |
| struct msmfb_data *data) |
| { |
| struct msm_fb_data_type *mfd = (struct msm_fb_data_type *) info->par; |
| |
| if (!mfd) |
| return -ENODEV; |
| |
| return mdss_mdp_wb_queue(mfd, data, true); |
| } |
| EXPORT_SYMBOL(msm_fb_writeback_queue_buffer); |
| |
| int msm_fb_writeback_dequeue_buffer(struct fb_info *info, |
| struct msmfb_data *data) |
| { |
| struct msm_fb_data_type *mfd = (struct msm_fb_data_type *) info->par; |
| |
| if (!mfd) |
| return -ENODEV; |
| |
| return mdss_mdp_wb_dequeue(mfd, data); |
| } |
| EXPORT_SYMBOL(msm_fb_writeback_dequeue_buffer); |
| |
| int msm_fb_writeback_stop(struct fb_info *info) |
| { |
| struct msm_fb_data_type *mfd = (struct msm_fb_data_type *) info->par; |
| |
| if (!mfd) |
| return -ENODEV; |
| |
| return mdss_mdp_wb_stop(mfd); |
| } |
| EXPORT_SYMBOL(msm_fb_writeback_stop); |
| |
| int msm_fb_writeback_init(struct fb_info *info) |
| { |
| struct msm_fb_data_type *mfd = (struct msm_fb_data_type *) info->par; |
| |
| if (!mfd) |
| return -ENODEV; |
| |
| return mdss_mdp_wb_init(mfd); |
| } |
| EXPORT_SYMBOL(msm_fb_writeback_init); |
| |
| int msm_fb_writeback_terminate(struct fb_info *info) |
| { |
| struct msm_fb_data_type *mfd = (struct msm_fb_data_type *) info->par; |
| |
| if (!mfd) |
| return -ENODEV; |
| |
| return mdss_mdp_wb_terminate(mfd); |
| } |
| EXPORT_SYMBOL(msm_fb_writeback_terminate); |
| |
| int msm_fb_get_iommu_domain(struct fb_info *info, int domain) |
| { |
| int mdss_domain; |
| switch (domain) { |
| case MDP_IOMMU_DOMAIN_CP: |
| mdss_domain = MDSS_IOMMU_DOMAIN_SECURE; |
| break; |
| case MDP_IOMMU_DOMAIN_NS: |
| mdss_domain = MDSS_IOMMU_DOMAIN_UNSECURE; |
| break; |
| default: |
| pr_err("Invalid mdp iommu domain (%d)\n", domain); |
| return -EINVAL; |
| } |
| return mdss_get_iommu_domain(mdss_domain); |
| } |
| EXPORT_SYMBOL(msm_fb_get_iommu_domain); |
| |
| int msm_fb_writeback_set_secure(struct fb_info *info, int enable) |
| { |
| struct msm_fb_data_type *mfd = (struct msm_fb_data_type *) info->par; |
| |
| if (!mfd) |
| return -ENODEV; |
| |
| return mdss_mdp_wb_set_secure(mfd, enable); |
| } |
| EXPORT_SYMBOL(msm_fb_writeback_set_secure); |