| /* |
| * Copyright (C) Fuzhou Rockchip Electronics Co.Ltd |
| * Author:Mark Yao <mark.yao@rock-chips.com> |
| * |
| * This software is licensed under the terms of the GNU General Public |
| * License version 2, as published by the Free Software Foundation, and |
| * may be copied, distributed, and modified under those terms. |
| * |
| * 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. |
| */ |
| |
| #include <linux/kernel.h> |
| #include <drm/drm.h> |
| #include <drm/drmP.h> |
| #include <drm/drm_atomic.h> |
| #include <drm/drm_fb_helper.h> |
| #include <drm/drm_crtc_helper.h> |
| #include <drm/drm_gem_framebuffer_helper.h> |
| |
| #include "rockchip_drm_drv.h" |
| #include "rockchip_drm_fb.h" |
| #include "rockchip_drm_gem.h" |
| #include "rockchip_drm_psr.h" |
| |
| static int rockchip_drm_fb_dirty(struct drm_framebuffer *fb, |
| struct drm_file *file, |
| unsigned int flags, unsigned int color, |
| struct drm_clip_rect *clips, |
| unsigned int num_clips) |
| { |
| rockchip_drm_psr_flush_all(fb->dev); |
| return 0; |
| } |
| |
| static const struct drm_framebuffer_funcs rockchip_drm_fb_funcs = { |
| .destroy = drm_gem_fb_destroy, |
| .create_handle = drm_gem_fb_create_handle, |
| .dirty = rockchip_drm_fb_dirty, |
| }; |
| |
| static struct drm_framebuffer * |
| rockchip_fb_alloc(struct drm_device *dev, const struct drm_mode_fb_cmd2 *mode_cmd, |
| struct drm_gem_object **obj, unsigned int num_planes) |
| { |
| struct drm_framebuffer *fb; |
| int ret; |
| int i; |
| |
| fb = kzalloc(sizeof(*fb), GFP_KERNEL); |
| if (!fb) |
| return ERR_PTR(-ENOMEM); |
| |
| drm_helper_mode_fill_fb_struct(dev, fb, mode_cmd); |
| |
| for (i = 0; i < num_planes; i++) |
| fb->obj[i] = obj[i]; |
| |
| ret = drm_framebuffer_init(dev, fb, &rockchip_drm_fb_funcs); |
| if (ret) { |
| DRM_DEV_ERROR(dev->dev, |
| "Failed to initialize framebuffer: %d\n", |
| ret); |
| kfree(fb); |
| return ERR_PTR(ret); |
| } |
| |
| return fb; |
| } |
| |
| static struct drm_framebuffer * |
| rockchip_user_fb_create(struct drm_device *dev, struct drm_file *file_priv, |
| const struct drm_mode_fb_cmd2 *mode_cmd) |
| { |
| struct drm_framebuffer *fb; |
| struct drm_gem_object *objs[ROCKCHIP_MAX_FB_BUFFER]; |
| struct drm_gem_object *obj; |
| unsigned int hsub; |
| unsigned int vsub; |
| int num_planes; |
| int ret; |
| int i; |
| |
| hsub = drm_format_horz_chroma_subsampling(mode_cmd->pixel_format); |
| vsub = drm_format_vert_chroma_subsampling(mode_cmd->pixel_format); |
| num_planes = min(drm_format_num_planes(mode_cmd->pixel_format), |
| ROCKCHIP_MAX_FB_BUFFER); |
| |
| for (i = 0; i < num_planes; i++) { |
| unsigned int width = mode_cmd->width / (i ? hsub : 1); |
| unsigned int height = mode_cmd->height / (i ? vsub : 1); |
| unsigned int min_size; |
| |
| obj = drm_gem_object_lookup(file_priv, mode_cmd->handles[i]); |
| if (!obj) { |
| DRM_DEV_ERROR(dev->dev, |
| "Failed to lookup GEM object\n"); |
| ret = -ENXIO; |
| goto err_gem_object_unreference; |
| } |
| |
| min_size = (height - 1) * mode_cmd->pitches[i] + |
| mode_cmd->offsets[i] + |
| width * drm_format_plane_cpp(mode_cmd->pixel_format, i); |
| |
| if (obj->size < min_size) { |
| drm_gem_object_put_unlocked(obj); |
| ret = -EINVAL; |
| goto err_gem_object_unreference; |
| } |
| objs[i] = obj; |
| } |
| |
| fb = rockchip_fb_alloc(dev, mode_cmd, objs, i); |
| if (IS_ERR(fb)) { |
| ret = PTR_ERR(fb); |
| goto err_gem_object_unreference; |
| } |
| |
| return fb; |
| |
| err_gem_object_unreference: |
| for (i--; i >= 0; i--) |
| drm_gem_object_put_unlocked(objs[i]); |
| return ERR_PTR(ret); |
| } |
| |
| static void |
| rockchip_drm_psr_inhibit_get_state(struct drm_atomic_state *state) |
| { |
| struct drm_crtc *crtc; |
| struct drm_crtc_state *crtc_state; |
| struct drm_encoder *encoder; |
| u32 encoder_mask = 0; |
| int i; |
| |
| for_each_old_crtc_in_state(state, crtc, crtc_state, i) { |
| encoder_mask |= crtc_state->encoder_mask; |
| encoder_mask |= crtc->state->encoder_mask; |
| } |
| |
| drm_for_each_encoder_mask(encoder, state->dev, encoder_mask) |
| rockchip_drm_psr_inhibit_get(encoder); |
| } |
| |
| static void |
| rockchip_drm_psr_inhibit_put_state(struct drm_atomic_state *state) |
| { |
| struct drm_crtc *crtc; |
| struct drm_crtc_state *crtc_state; |
| struct drm_encoder *encoder; |
| u32 encoder_mask = 0; |
| int i; |
| |
| for_each_old_crtc_in_state(state, crtc, crtc_state, i) { |
| encoder_mask |= crtc_state->encoder_mask; |
| encoder_mask |= crtc->state->encoder_mask; |
| } |
| |
| drm_for_each_encoder_mask(encoder, state->dev, encoder_mask) |
| rockchip_drm_psr_inhibit_put(encoder); |
| } |
| |
| static void |
| rockchip_atomic_helper_commit_tail_rpm(struct drm_atomic_state *old_state) |
| { |
| struct drm_device *dev = old_state->dev; |
| |
| rockchip_drm_psr_inhibit_get_state(old_state); |
| |
| drm_atomic_helper_commit_modeset_disables(dev, old_state); |
| |
| drm_atomic_helper_commit_modeset_enables(dev, old_state); |
| |
| drm_atomic_helper_commit_planes(dev, old_state, |
| DRM_PLANE_COMMIT_ACTIVE_ONLY); |
| |
| rockchip_drm_psr_inhibit_put_state(old_state); |
| |
| drm_atomic_helper_commit_hw_done(old_state); |
| |
| drm_atomic_helper_wait_for_vblanks(dev, old_state); |
| |
| drm_atomic_helper_cleanup_planes(dev, old_state); |
| } |
| |
| static const struct drm_mode_config_helper_funcs rockchip_mode_config_helpers = { |
| .atomic_commit_tail = rockchip_atomic_helper_commit_tail_rpm, |
| }; |
| |
| static const struct drm_mode_config_funcs rockchip_drm_mode_config_funcs = { |
| .fb_create = rockchip_user_fb_create, |
| .output_poll_changed = drm_fb_helper_output_poll_changed, |
| .atomic_check = drm_atomic_helper_check, |
| .atomic_commit = drm_atomic_helper_commit, |
| }; |
| |
| struct drm_framebuffer * |
| rockchip_drm_framebuffer_init(struct drm_device *dev, |
| const struct drm_mode_fb_cmd2 *mode_cmd, |
| struct drm_gem_object *obj) |
| { |
| struct drm_framebuffer *fb; |
| |
| fb = rockchip_fb_alloc(dev, mode_cmd, &obj, 1); |
| if (IS_ERR(fb)) |
| return ERR_CAST(fb); |
| |
| return fb; |
| } |
| |
| void rockchip_drm_mode_config_init(struct drm_device *dev) |
| { |
| dev->mode_config.min_width = 0; |
| dev->mode_config.min_height = 0; |
| |
| /* |
| * set max width and height as default value(4096x4096). |
| * this value would be used to check framebuffer size limitation |
| * at drm_mode_addfb(). |
| */ |
| dev->mode_config.max_width = 4096; |
| dev->mode_config.max_height = 4096; |
| |
| dev->mode_config.funcs = &rockchip_drm_mode_config_funcs; |
| dev->mode_config.helper_private = &rockchip_mode_config_helpers; |
| } |