Commit 3314fdf8 authored by Dave Airlie's avatar Dave Airlie

Merge branch 'drm-fb-helper' of git://people.freedesktop.org/~danvet/drm into drm-next

This is the drm fb helper cleanup, mostly motivated by strange things I've
seen in my locking rework and the i915 modeset revamp. Compared to the
original submission I've reinstated the setup flexibility you'd like to
retain, kerneldoc has been reviewed by Laurent Pinchart and Rob Clark
reviewed the code changes.

Quick overview of the changes:
- Cleaned-up library interface for drivers using the fb helper, also
  simplified the fb allocation callback since no driver supported
  reallocating the fb on-the-fly. And the fbdev/fbcon code keeps pointers
  to the old mapping around anyway, so reallocating backing storage will
  be much more work.
- No longer call the crtc helper "disable everything" function at init
  time, but allow drivers to do so. Motivated by i915's fastboot effort
  and allows us to drop a bunch of noop dummy functions just to avoid
  calling NULL function pointers from i915.ko.
- Properly clear old state when doing modeset calls, the fb helper left
  some old modes in there and unconditionally set an fb (even when
  disabling a crtc). The crtc helpers didn't care, but i915 modeset code
  can now drop a few special cases.
- Full kerneldoc for the fb helper. Yay!
- My version of the "don't sleep in panic ->unblank calls". The patch is
  already in -mm, I guess Andrew can drop it as soon as this pull lands in
  drm-next.

* 'drm-fb-helper' of git://people.freedesktop.org/~danvet/drm:
  drm/fb-helper: remove unused members of struct drm_fb_helper
  drm/fb-helper: don't sleep for screen unblank when an oopps is in progress
  drm/fb-helper: improve kerneldoc
  drm/<drivers>: simplify ->fb_probe callback
  drm/fb-helper: streamline drm_fb_helper_single_fb_probe
  drm/fb-helper: directly call set_par from the hotplug handler
  drm/fb-helper: fixup set_config semantics
  drm/i915: rip out helper->disable noop functions
  drm/fb-helper: don't disable everything in initial_config
  drm/tegra: don't set up initial fbcon config twice
  drm/fb-helper: unexport drm_fb_helper_single_fb_probe
  drm/fb-helper: unexport drm_fb_helper_panic
  drm/fb-helper: kill drm_fb_helper_restore
  drm: review locking for drm_fb_helper_restore_fbdev_mode
parents f934ec8c a065b46a
...@@ -2143,6 +2143,7 @@ void intel_crt_init(struct drm_device *dev) ...@@ -2143,6 +2143,7 @@ void intel_crt_init(struct drm_device *dev)
<title>fbdev Helper Functions Reference</title> <title>fbdev Helper Functions Reference</title>
!Pdrivers/gpu/drm/drm_fb_helper.c fbdev helpers !Pdrivers/gpu/drm/drm_fb_helper.c fbdev helpers
!Edrivers/gpu/drm/drm_fb_helper.c !Edrivers/gpu/drm/drm_fb_helper.c
!Iinclude/drm/drm_fb_helper.h
</sect2> </sect2>
<sect2> <sect2>
<title>Display Port Helper Functions Reference</title> <title>Display Port Helper Functions Reference</title>
......
...@@ -40,6 +40,7 @@ ...@@ -40,6 +40,7 @@
#include <drm/drmP.h> #include <drm/drmP.h>
#include <drm/drm_crtc.h> #include <drm/drm_crtc.h>
#include <drm/drm_fb_helper.h> #include <drm/drm_fb_helper.h>
#include <drm/drm_crtc_helper.h>
#include "ast_drv.h" #include "ast_drv.h"
static void ast_dirty_update(struct ast_fbdev *afbdev, static void ast_dirty_update(struct ast_fbdev *afbdev,
...@@ -145,9 +146,10 @@ static int astfb_create_object(struct ast_fbdev *afbdev, ...@@ -145,9 +146,10 @@ static int astfb_create_object(struct ast_fbdev *afbdev,
return ret; return ret;
} }
static int astfb_create(struct ast_fbdev *afbdev, static int astfb_create(struct drm_fb_helper *helper,
struct drm_fb_helper_surface_size *sizes) struct drm_fb_helper_surface_size *sizes)
{ {
struct ast_fbdev *afbdev = (struct ast_fbdev *)helper;
struct drm_device *dev = afbdev->helper.dev; struct drm_device *dev = afbdev->helper.dev;
struct drm_mode_fb_cmd2 mode_cmd; struct drm_mode_fb_cmd2 mode_cmd;
struct drm_framebuffer *fb; struct drm_framebuffer *fb;
...@@ -248,26 +250,10 @@ static void ast_fb_gamma_get(struct drm_crtc *crtc, u16 *red, u16 *green, ...@@ -248,26 +250,10 @@ static void ast_fb_gamma_get(struct drm_crtc *crtc, u16 *red, u16 *green,
*blue = ast_crtc->lut_b[regno] << 8; *blue = ast_crtc->lut_b[regno] << 8;
} }
static int ast_find_or_create_single(struct drm_fb_helper *helper,
struct drm_fb_helper_surface_size *sizes)
{
struct ast_fbdev *afbdev = (struct ast_fbdev *)helper;
int new_fb = 0;
int ret;
if (!helper->fb) {
ret = astfb_create(afbdev, sizes);
if (ret)
return ret;
new_fb = 1;
}
return new_fb;
}
static struct drm_fb_helper_funcs ast_fb_helper_funcs = { static struct drm_fb_helper_funcs ast_fb_helper_funcs = {
.gamma_set = ast_fb_gamma_set, .gamma_set = ast_fb_gamma_set,
.gamma_get = ast_fb_gamma_get, .gamma_get = ast_fb_gamma_get,
.fb_probe = ast_find_or_create_single, .fb_probe = astfb_create,
}; };
static void ast_fbdev_destroy(struct drm_device *dev, static void ast_fbdev_destroy(struct drm_device *dev,
...@@ -314,6 +300,10 @@ int ast_fbdev_init(struct drm_device *dev) ...@@ -314,6 +300,10 @@ int ast_fbdev_init(struct drm_device *dev)
} }
drm_fb_helper_single_add_all_connectors(&afbdev->helper); drm_fb_helper_single_add_all_connectors(&afbdev->helper);
/* disable all the possible outputs/crtcs before entering KMS mode */
drm_helper_disable_unused_functions(dev);
drm_fb_helper_initial_config(&afbdev->helper, 32); drm_fb_helper_initial_config(&afbdev->helper, 32);
return 0; return 0;
} }
......
...@@ -11,6 +11,7 @@ ...@@ -11,6 +11,7 @@
#include <linux/module.h> #include <linux/module.h>
#include <drm/drmP.h> #include <drm/drmP.h>
#include <drm/drm_fb_helper.h> #include <drm/drm_fb_helper.h>
#include <drm/drm_crtc_helper.h>
#include <linux/fb.h> #include <linux/fb.h>
...@@ -120,9 +121,10 @@ static int cirrusfb_create_object(struct cirrus_fbdev *afbdev, ...@@ -120,9 +121,10 @@ static int cirrusfb_create_object(struct cirrus_fbdev *afbdev,
return ret; return ret;
} }
static int cirrusfb_create(struct cirrus_fbdev *gfbdev, static int cirrusfb_create(struct drm_fb_helper *helper,
struct drm_fb_helper_surface_size *sizes) struct drm_fb_helper_surface_size *sizes)
{ {
struct cirrus_fbdev *gfbdev = (struct cirrus_fbdev *)helper;
struct drm_device *dev = gfbdev->helper.dev; struct drm_device *dev = gfbdev->helper.dev;
struct cirrus_device *cdev = gfbdev->helper.dev->dev_private; struct cirrus_device *cdev = gfbdev->helper.dev->dev_private;
struct fb_info *info; struct fb_info *info;
...@@ -219,23 +221,6 @@ static int cirrusfb_create(struct cirrus_fbdev *gfbdev, ...@@ -219,23 +221,6 @@ static int cirrusfb_create(struct cirrus_fbdev *gfbdev,
return ret; return ret;
} }
static int cirrus_fb_find_or_create_single(struct drm_fb_helper *helper,
struct drm_fb_helper_surface_size
*sizes)
{
struct cirrus_fbdev *gfbdev = (struct cirrus_fbdev *)helper;
int new_fb = 0;
int ret;
if (!helper->fb) {
ret = cirrusfb_create(gfbdev, sizes);
if (ret)
return ret;
new_fb = 1;
}
return new_fb;
}
static int cirrus_fbdev_destroy(struct drm_device *dev, static int cirrus_fbdev_destroy(struct drm_device *dev,
struct cirrus_fbdev *gfbdev) struct cirrus_fbdev *gfbdev)
{ {
...@@ -267,7 +252,7 @@ static int cirrus_fbdev_destroy(struct drm_device *dev, ...@@ -267,7 +252,7 @@ static int cirrus_fbdev_destroy(struct drm_device *dev,
static struct drm_fb_helper_funcs cirrus_fb_helper_funcs = { static struct drm_fb_helper_funcs cirrus_fb_helper_funcs = {
.gamma_set = cirrus_crtc_fb_gamma_set, .gamma_set = cirrus_crtc_fb_gamma_set,
.gamma_get = cirrus_crtc_fb_gamma_get, .gamma_get = cirrus_crtc_fb_gamma_get,
.fb_probe = cirrus_fb_find_or_create_single, .fb_probe = cirrusfb_create,
}; };
int cirrus_fbdev_init(struct cirrus_device *cdev) int cirrus_fbdev_init(struct cirrus_device *cdev)
...@@ -291,6 +276,9 @@ int cirrus_fbdev_init(struct cirrus_device *cdev) ...@@ -291,6 +276,9 @@ int cirrus_fbdev_init(struct cirrus_device *cdev)
return ret; return ret;
} }
drm_fb_helper_single_add_all_connectors(&gfbdev->helper); drm_fb_helper_single_add_all_connectors(&gfbdev->helper);
/* disable all the possible outputs/crtcs before entering KMS mode */
drm_helper_disable_unused_functions(cdev->dev);
drm_fb_helper_initial_config(&gfbdev->helper, bpp_sel); drm_fb_helper_initial_config(&gfbdev->helper, bpp_sel);
return 0; return 0;
......
...@@ -68,9 +68,23 @@ void drm_modeset_unlock_all(struct drm_device *dev) ...@@ -68,9 +68,23 @@ void drm_modeset_unlock_all(struct drm_device *dev)
mutex_unlock(&dev->mode_config.mutex); mutex_unlock(&dev->mode_config.mutex);
} }
EXPORT_SYMBOL(drm_modeset_unlock_all); EXPORT_SYMBOL(drm_modeset_unlock_all);
/**
* drm_warn_on_modeset_not_all_locked - check that all modeset locks are locked
* @dev: device
*/
void drm_warn_on_modeset_not_all_locked(struct drm_device *dev)
{
struct drm_crtc *crtc;
list_for_each_entry(crtc, &dev->mode_config.crtc_list, head)
WARN_ON(!mutex_is_locked(&crtc->mutex));
WARN_ON(!mutex_is_locked(&dev->mode_config.mutex));
}
EXPORT_SYMBOL(drm_warn_on_modeset_not_all_locked);
/* Avoid boilerplate. I'm tired of typing. */ /* Avoid boilerplate. I'm tired of typing. */
#define DRM_ENUM_NAME_FN(fnname, list) \ #define DRM_ENUM_NAME_FN(fnname, list) \
char *fnname(int val) \ char *fnname(int val) \
......
...@@ -275,23 +275,8 @@ static int drm_fbdev_cma_create(struct drm_fb_helper *helper, ...@@ -275,23 +275,8 @@ static int drm_fbdev_cma_create(struct drm_fb_helper *helper,
return ret; return ret;
} }
static int drm_fbdev_cma_probe(struct drm_fb_helper *helper,
struct drm_fb_helper_surface_size *sizes)
{
int ret = 0;
if (!helper->fb) {
ret = drm_fbdev_cma_create(helper, sizes);
if (ret < 0)
return ret;
ret = 1;
}
return ret;
}
static struct drm_fb_helper_funcs drm_fb_cma_helper_funcs = { static struct drm_fb_helper_funcs drm_fb_cma_helper_funcs = {
.fb_probe = drm_fbdev_cma_probe, .fb_probe = drm_fbdev_cma_create,
}; };
/** /**
...@@ -333,6 +318,9 @@ struct drm_fbdev_cma *drm_fbdev_cma_init(struct drm_device *dev, ...@@ -333,6 +318,9 @@ struct drm_fbdev_cma *drm_fbdev_cma_init(struct drm_device *dev,
} }
/* disable all the possible outputs/crtcs before entering KMS mode */
drm_helper_disable_unused_functions(dev);
ret = drm_fb_helper_initial_config(helper, preferred_bpp); ret = drm_fb_helper_initial_config(helper, preferred_bpp);
if (ret < 0) { if (ret < 0) {
dev_err(dev->dev, "Failed to set inital hw configuration.\n"); dev_err(dev->dev, "Failed to set inital hw configuration.\n");
...@@ -389,8 +377,10 @@ EXPORT_SYMBOL_GPL(drm_fbdev_cma_fini); ...@@ -389,8 +377,10 @@ EXPORT_SYMBOL_GPL(drm_fbdev_cma_fini);
*/ */
void drm_fbdev_cma_restore_mode(struct drm_fbdev_cma *fbdev_cma) void drm_fbdev_cma_restore_mode(struct drm_fbdev_cma *fbdev_cma)
{ {
drm_modeset_lock_all(dev);
if (fbdev_cma) if (fbdev_cma)
drm_fb_helper_restore_fbdev_mode(&fbdev_cma->fb_helper); drm_fb_helper_restore_fbdev_mode(&fbdev_cma->fb_helper);
drm_modeset_unlock_all(dev);
} }
EXPORT_SYMBOL_GPL(drm_fbdev_cma_restore_mode); EXPORT_SYMBOL_GPL(drm_fbdev_cma_restore_mode);
......
This diff is collapsed.
...@@ -226,36 +226,8 @@ static int exynos_drm_fbdev_create(struct drm_fb_helper *helper, ...@@ -226,36 +226,8 @@ static int exynos_drm_fbdev_create(struct drm_fb_helper *helper,
return ret; return ret;
} }
static int exynos_drm_fbdev_probe(struct drm_fb_helper *helper,
struct drm_fb_helper_surface_size *sizes)
{
int ret = 0;
DRM_DEBUG_KMS("%s\n", __FILE__);
/*
* with !helper->fb, it means that this funcion is called first time
* and after that, the helper->fb would be used as clone mode.
*/
if (!helper->fb) {
ret = exynos_drm_fbdev_create(helper, sizes);
if (ret < 0) {
DRM_ERROR("failed to create fbdev.\n");
return ret;
}
/*
* fb_helper expects a value more than 1 if succeed
* because register_framebuffer() should be called.
*/
ret = 1;
}
return ret;
}
static struct drm_fb_helper_funcs exynos_drm_fb_helper_funcs = { static struct drm_fb_helper_funcs exynos_drm_fb_helper_funcs = {
.fb_probe = exynos_drm_fbdev_probe, .fb_probe = exynos_drm_fbdev_create,
}; };
int exynos_drm_fbdev_init(struct drm_device *dev) int exynos_drm_fbdev_init(struct drm_device *dev)
...@@ -295,6 +267,9 @@ int exynos_drm_fbdev_init(struct drm_device *dev) ...@@ -295,6 +267,9 @@ int exynos_drm_fbdev_init(struct drm_device *dev)
} }
/* disable all the possible outputs/crtcs before entering KMS mode */
drm_helper_disable_unused_functions(dev);
ret = drm_fb_helper_initial_config(helper, PREFERRED_BPP); ret = drm_fb_helper_initial_config(helper, PREFERRED_BPP);
if (ret < 0) { if (ret < 0) {
DRM_ERROR("failed to set up hw configuration.\n"); DRM_ERROR("failed to set up hw configuration.\n");
...@@ -376,5 +351,7 @@ void exynos_drm_fbdev_restore_mode(struct drm_device *dev) ...@@ -376,5 +351,7 @@ void exynos_drm_fbdev_restore_mode(struct drm_device *dev)
if (!private || !private->fb_helper) if (!private || !private->fb_helper)
return; return;
drm_modeset_lock_all(dev);
drm_fb_helper_restore_fbdev_mode(private->fb_helper); drm_fb_helper_restore_fbdev_mode(private->fb_helper);
drm_modeset_unlock_all(dev);
} }
...@@ -545,9 +545,7 @@ static int psbfb_probe(struct drm_fb_helper *helper, ...@@ -545,9 +545,7 @@ static int psbfb_probe(struct drm_fb_helper *helper,
struct psb_fbdev *psb_fbdev = (struct psb_fbdev *)helper; struct psb_fbdev *psb_fbdev = (struct psb_fbdev *)helper;
struct drm_device *dev = psb_fbdev->psb_fb_helper.dev; struct drm_device *dev = psb_fbdev->psb_fb_helper.dev;
struct drm_psb_private *dev_priv = dev->dev_private; struct drm_psb_private *dev_priv = dev->dev_private;
int new_fb = 0;
int bytespp; int bytespp;
int ret;
bytespp = sizes->surface_bpp / 8; bytespp = sizes->surface_bpp / 8;
if (bytespp == 3) /* no 24bit packed */ if (bytespp == 3) /* no 24bit packed */
...@@ -562,13 +560,7 @@ static int psbfb_probe(struct drm_fb_helper *helper, ...@@ -562,13 +560,7 @@ static int psbfb_probe(struct drm_fb_helper *helper,
sizes->surface_depth = 16; sizes->surface_depth = 16;
} }
if (!helper->fb) { return psbfb_create(psb_fbdev, sizes);
ret = psbfb_create(psb_fbdev, sizes);
if (ret)
return ret;
new_fb = 1;
}
return new_fb;
} }
static struct drm_fb_helper_funcs psb_fb_helper_funcs = { static struct drm_fb_helper_funcs psb_fb_helper_funcs = {
...@@ -616,6 +608,10 @@ int psb_fbdev_init(struct drm_device *dev) ...@@ -616,6 +608,10 @@ int psb_fbdev_init(struct drm_device *dev)
INTELFB_CONN_LIMIT); INTELFB_CONN_LIMIT);
drm_fb_helper_single_add_all_connectors(&fbdev->psb_fb_helper); drm_fb_helper_single_add_all_connectors(&fbdev->psb_fb_helper);
/* disable all the possible outputs/crtcs before entering KMS mode */
drm_helper_disable_unused_functions(dev);
drm_fb_helper_initial_config(&fbdev->psb_fb_helper, 32); drm_fb_helper_initial_config(&fbdev->psb_fb_helper, 32);
return 0; return 0;
} }
......
...@@ -685,7 +685,6 @@ static void intel_crt_reset(struct drm_connector *connector) ...@@ -685,7 +685,6 @@ static void intel_crt_reset(struct drm_connector *connector)
static const struct drm_encoder_helper_funcs crt_encoder_funcs = { static const struct drm_encoder_helper_funcs crt_encoder_funcs = {
.mode_fixup = intel_crt_mode_fixup, .mode_fixup = intel_crt_mode_fixup,
.mode_set = intel_crt_mode_set, .mode_set = intel_crt_mode_set,
.disable = intel_encoder_noop,
}; };
static const struct drm_connector_funcs intel_crt_connector_funcs = { static const struct drm_connector_funcs intel_crt_connector_funcs = {
......
...@@ -1481,7 +1481,6 @@ static const struct drm_encoder_funcs intel_ddi_funcs = { ...@@ -1481,7 +1481,6 @@ static const struct drm_encoder_funcs intel_ddi_funcs = {
static const struct drm_encoder_helper_funcs intel_ddi_helper_funcs = { static const struct drm_encoder_helper_funcs intel_ddi_helper_funcs = {
.mode_fixup = intel_ddi_mode_fixup, .mode_fixup = intel_ddi_mode_fixup,
.mode_set = intel_ddi_mode_set, .mode_set = intel_ddi_mode_set,
.disable = intel_encoder_noop,
}; };
void intel_ddi_init(struct drm_device *dev, enum port port) void intel_ddi_init(struct drm_device *dev, enum port port)
......
...@@ -3718,10 +3718,6 @@ void intel_crtc_update_dpms(struct drm_crtc *crtc) ...@@ -3718,10 +3718,6 @@ void intel_crtc_update_dpms(struct drm_crtc *crtc)
intel_crtc_update_sarea(crtc, enable); intel_crtc_update_sarea(crtc, enable);
} }
static void intel_crtc_noop(struct drm_crtc *crtc)
{
}
static void intel_crtc_disable(struct drm_crtc *crtc) static void intel_crtc_disable(struct drm_crtc *crtc)
{ {
struct drm_device *dev = crtc->dev; struct drm_device *dev = crtc->dev;
...@@ -3770,10 +3766,6 @@ void intel_modeset_disable(struct drm_device *dev) ...@@ -3770,10 +3766,6 @@ void intel_modeset_disable(struct drm_device *dev)
} }
} }
void intel_encoder_noop(struct drm_encoder *encoder)
{
}
void intel_encoder_destroy(struct drm_encoder *encoder) void intel_encoder_destroy(struct drm_encoder *encoder)
{ {
struct intel_encoder *intel_encoder = to_intel_encoder(encoder); struct intel_encoder *intel_encoder = to_intel_encoder(encoder);
...@@ -7277,7 +7269,6 @@ static int intel_crtc_page_flip(struct drm_crtc *crtc, ...@@ -7277,7 +7269,6 @@ static int intel_crtc_page_flip(struct drm_crtc *crtc,
static struct drm_crtc_helper_funcs intel_helper_funcs = { static struct drm_crtc_helper_funcs intel_helper_funcs = {
.mode_set_base_atomic = intel_pipe_set_base_atomic, .mode_set_base_atomic = intel_pipe_set_base_atomic,
.load_lut = intel_crtc_load_lut, .load_lut = intel_crtc_load_lut,
.disable = intel_crtc_noop,
}; };
bool intel_encoder_check_is_cloned(struct intel_encoder *encoder) bool intel_encoder_check_is_cloned(struct intel_encoder *encoder)
...@@ -7987,14 +7978,9 @@ static int intel_crtc_set_config(struct drm_mode_set *set) ...@@ -7987,14 +7978,9 @@ static int intel_crtc_set_config(struct drm_mode_set *set)
BUG_ON(!set->crtc); BUG_ON(!set->crtc);
BUG_ON(!set->crtc->helper_private); BUG_ON(!set->crtc->helper_private);
if (!set->mode) /* Enforce sane interface api - has been abused by the fb helper. */
set->fb = NULL; BUG_ON(!set->mode && set->fb);
BUG_ON(set->fb && set->num_connectors == 0);
/* The fb helper likes to play gross jokes with ->mode_set_config.
* Unfortunately the crtc helper doesn't do much at all for this case,
* so we have to cope with this madness until the fb helper is fixed up. */
if (set->fb && set->num_connectors == 0)
return 0;
if (set->fb) { if (set->fb) {
DRM_DEBUG_KMS("[CRTC:%d] [FB:%d] #connectors=%d (x y) (%i %i)\n", DRM_DEBUG_KMS("[CRTC:%d] [FB:%d] #connectors=%d (x y) (%i %i)\n",
......
...@@ -2561,7 +2561,6 @@ void intel_dp_encoder_destroy(struct drm_encoder *encoder) ...@@ -2561,7 +2561,6 @@ void intel_dp_encoder_destroy(struct drm_encoder *encoder)
static const struct drm_encoder_helper_funcs intel_dp_helper_funcs = { static const struct drm_encoder_helper_funcs intel_dp_helper_funcs = {
.mode_fixup = intel_dp_mode_fixup, .mode_fixup = intel_dp_mode_fixup,
.mode_set = intel_dp_mode_set, .mode_set = intel_dp_mode_set,
.disable = intel_encoder_noop,
}; };
static const struct drm_connector_funcs intel_dp_connector_funcs = { static const struct drm_connector_funcs intel_dp_connector_funcs = {
......
...@@ -521,7 +521,6 @@ extern void intel_modeset_disable(struct drm_device *dev); ...@@ -521,7 +521,6 @@ extern void intel_modeset_disable(struct drm_device *dev);
extern void intel_crtc_restore_mode(struct drm_crtc *crtc); extern void intel_crtc_restore_mode(struct drm_crtc *crtc);
extern void intel_crtc_load_lut(struct drm_crtc *crtc); extern void intel_crtc_load_lut(struct drm_crtc *crtc);
extern void intel_crtc_update_dpms(struct drm_crtc *crtc); extern void intel_crtc_update_dpms(struct drm_crtc *crtc);
extern void intel_encoder_noop(struct drm_encoder *encoder);
extern void intel_encoder_destroy(struct drm_encoder *encoder); extern void intel_encoder_destroy(struct drm_encoder *encoder);
extern void intel_encoder_dpms(struct intel_encoder *encoder, int mode); extern void intel_encoder_dpms(struct intel_encoder *encoder, int mode);
extern bool intel_encoder_check_is_cloned(struct intel_encoder *encoder); extern bool intel_encoder_check_is_cloned(struct intel_encoder *encoder);
......
...@@ -345,7 +345,6 @@ static void intel_dvo_destroy(struct drm_connector *connector) ...@@ -345,7 +345,6 @@ static void intel_dvo_destroy(struct drm_connector *connector)
static const struct drm_encoder_helper_funcs intel_dvo_helper_funcs = { static const struct drm_encoder_helper_funcs intel_dvo_helper_funcs = {
.mode_fixup = intel_dvo_mode_fixup, .mode_fixup = intel_dvo_mode_fixup,
.mode_set = intel_dvo_mode_set, .mode_set = intel_dvo_mode_set,
.disable = intel_encoder_noop,
}; };
static const struct drm_connector_funcs intel_dvo_connector_funcs = { static const struct drm_connector_funcs intel_dvo_connector_funcs = {
......
...@@ -57,9 +57,10 @@ static struct fb_ops intelfb_ops = { ...@@ -57,9 +57,10 @@ static struct fb_ops intelfb_ops = {
.fb_debug_leave = drm_fb_helper_debug_leave, .fb_debug_leave = drm_fb_helper_debug_leave,
}; };
static int intelfb_create(struct intel_fbdev *ifbdev, static int intelfb_create(struct drm_fb_helper *helper,
struct drm_fb_helper_surface_size *sizes) struct drm_fb_helper_surface_size *sizes)
{ {
struct intel_fbdev *ifbdev = (struct intel_fbdev *)helper;
struct drm_device *dev = ifbdev->helper.dev; struct drm_device *dev = ifbdev->helper.dev;
struct drm_i915_private *dev_priv = dev->dev_private; struct drm_i915_private *dev_priv = dev->dev_private;
struct fb_info *info; struct fb_info *info;
...@@ -181,26 +182,10 @@ static int intelfb_create(struct intel_fbdev *ifbdev, ...@@ -181,26 +182,10 @@ static int intelfb_create(struct intel_fbdev *ifbdev,
return ret; return ret;
} }
static int intel_fb_find_or_create_single(struct drm_fb_helper *helper,
struct drm_fb_helper_surface_size *sizes)
{
struct intel_fbdev *ifbdev = (struct intel_fbdev *)helper;
int new_fb = 0;
int ret;
if (!helper->fb) {
ret = intelfb_create(ifbdev, sizes);
if (ret)
return ret;
new_fb = 1;
}
return new_fb;
}
static struct drm_fb_helper_funcs intel_fb_helper_funcs = { static struct drm_fb_helper_funcs intel_fb_helper_funcs = {
.gamma_set = intel_crtc_fb_gamma_set, .gamma_set = intel_crtc_fb_gamma_set,
.gamma_get = intel_crtc_fb_gamma_get, .gamma_get = intel_crtc_fb_gamma_get,
.fb_probe = intel_fb_find_or_create_single, .fb_probe = intelfb_create,
}; };
static void intel_fbdev_destroy(struct drm_device *dev, static void intel_fbdev_destroy(struct drm_device *dev,
......
...@@ -972,7 +972,6 @@ static void intel_hdmi_destroy(struct drm_connector *connector) ...@@ -972,7 +972,6 @@ static void intel_hdmi_destroy(struct drm_connector *connector)
static const struct drm_encoder_helper_funcs intel_hdmi_helper_funcs = { static const struct drm_encoder_helper_funcs intel_hdmi_helper_funcs = {
.mode_fixup = intel_hdmi_mode_fixup, .mode_fixup = intel_hdmi_mode_fixup,
.mode_set = intel_hdmi_mode_set, .mode_set = intel_hdmi_mode_set,
.disable = intel_encoder_noop,
}; };
static const struct drm_connector_funcs intel_hdmi_connector_funcs = { static const struct drm_connector_funcs intel_hdmi_connector_funcs = {
......
...@@ -656,7 +656,6 @@ static int intel_lvds_set_property(struct drm_connector *connector, ...@@ -656,7 +656,6 @@ static int intel_lvds_set_property(struct drm_connector *connector,
static const struct drm_encoder_helper_funcs intel_lvds_helper_funcs = { static const struct drm_encoder_helper_funcs intel_lvds_helper_funcs = {
.mode_fixup = intel_lvds_mode_fixup, .mode_fixup = intel_lvds_mode_fixup,
.mode_set = intel_lvds_mode_set, .mode_set = intel_lvds_mode_set,
.disable = intel_encoder_noop,
}; };
static const struct drm_connector_helper_funcs intel_lvds_connector_helper_funcs = { static const struct drm_connector_helper_funcs intel_lvds_connector_helper_funcs = {
......
...@@ -2043,7 +2043,6 @@ intel_sdvo_set_property(struct drm_connector *connector, ...@@ -2043,7 +2043,6 @@ intel_sdvo_set_property(struct drm_connector *connector,
static const struct drm_encoder_helper_funcs intel_sdvo_helper_funcs = { static const struct drm_encoder_helper_funcs intel_sdvo_helper_funcs = {
.mode_fixup = intel_sdvo_mode_fixup, .mode_fixup = intel_sdvo_mode_fixup,
.mode_set = intel_sdvo_mode_set, .mode_set = intel_sdvo_mode_set,
.disable = intel_encoder_noop,
}; };
static const struct drm_connector_funcs intel_sdvo_connector_funcs = { static const struct drm_connector_funcs intel_sdvo_connector_funcs = {
......
...@@ -1487,7 +1487,6 @@ intel_tv_set_property(struct drm_connector *connector, struct drm_property *prop ...@@ -1487,7 +1487,6 @@ intel_tv_set_property(struct drm_connector *connector, struct drm_property *prop
static const struct drm_encoder_helper_funcs intel_tv_helper_funcs = { static const struct drm_encoder_helper_funcs intel_tv_helper_funcs = {
.mode_fixup = intel_tv_mode_fixup, .mode_fixup = intel_tv_mode_fixup,
.mode_set = intel_tv_mode_set, .mode_set = intel_tv_mode_set,
.disable = intel_encoder_noop,
}; };
static const struct drm_connector_funcs intel_tv_connector_funcs = { static const struct drm_connector_funcs intel_tv_connector_funcs = {
......
...@@ -13,6 +13,7 @@ ...@@ -13,6 +13,7 @@
#include <linux/module.h> #include <linux/module.h>
#include <drm/drmP.h> #include <drm/drmP.h>
#include <drm/drm_fb_helper.h> #include <drm/drm_fb_helper.h>
#include <drm/drm_crtc_helper.h>
#include <linux/fb.h> #include <linux/fb.h>
...@@ -120,9 +121,10 @@ static int mgag200fb_create_object(struct mga_fbdev *afbdev, ...@@ -120,9 +121,10 @@ static int mgag200fb_create_object(struct mga_fbdev *afbdev,
return ret; return ret;
} }
static int mgag200fb_create(struct mga_fbdev *mfbdev, static int mgag200fb_create(struct drm_fb_helper *helper,
struct drm_fb_helper_surface_size *sizes) struct drm_fb_helper_surface_size *sizes)
{ {
struct mga_fbdev *mfbdev = (struct mga_fbdev *)helper;
struct drm_device *dev = mfbdev->helper.dev; struct drm_device *dev = mfbdev->helper.dev;
struct drm_mode_fb_cmd2 mode_cmd; struct drm_mode_fb_cmd2 mode_cmd;
struct mga_device *mdev = dev->dev_private; struct mga_device *mdev = dev->dev_private;
...@@ -209,23 +211,6 @@ static int mgag200fb_create(struct mga_fbdev *mfbdev, ...@@ -209,23 +211,6 @@ static int mgag200fb_create(struct mga_fbdev *mfbdev,
return ret; return ret;
} }
static int mga_fb_find_or_create_single(struct drm_fb_helper *helper,
struct drm_fb_helper_surface_size
*sizes)
{
struct mga_fbdev *mfbdev = (struct mga_fbdev *)helper;
int new_fb = 0;
int ret;
if (!helper->fb) {
ret = mgag200fb_create(mfbdev, sizes);
if (ret)
return ret;
new_fb = 1;
}
return new_fb;
}
static int mga_fbdev_destroy(struct drm_device *dev, static int mga_fbdev_destroy(struct drm_device *dev,
struct mga_fbdev *mfbdev) struct mga_fbdev *mfbdev)
{ {
...@@ -256,7 +241,7 @@ static int mga_fbdev_destroy(struct drm_device *dev, ...@@ -256,7 +241,7 @@ static int mga_fbdev_destroy(struct drm_device *dev,
static struct drm_fb_helper_funcs mga_fb_helper_funcs = { static struct drm_fb_helper_funcs mga_fb_helper_funcs = {
.gamma_set = mga_crtc_fb_gamma_set, .gamma_set = mga_crtc_fb_gamma_set,
.gamma_get = mga_crtc_fb_gamma_get, .gamma_get = mga_crtc_fb_gamma_get,
.fb_probe = mga_fb_find_or_create_single, .fb_probe = mgag200fb_create,
}; };
int mgag200_fbdev_init(struct mga_device *mdev) int mgag200_fbdev_init(struct mga_device *mdev)
...@@ -278,6 +263,10 @@ int mgag200_fbdev_init(struct mga_device *mdev) ...@@ -278,6 +263,10 @@ int mgag200_fbdev_init(struct mga_device *mdev)
return ret; return ret;
} }
drm_fb_helper_single_add_all_connectors(&mfbdev->helper); drm_fb_helper_single_add_all_connectors(&mfbdev->helper);
/* disable all the possible outputs/crtcs before entering KMS mode */
drm_helper_disable_unused_functions(mdev->dev);
drm_fb_helper_initial_config(&mfbdev->helper, 32); drm_fb_helper_initial_config(&mfbdev->helper, 32);
return 0; return 0;
......
...@@ -251,9 +251,10 @@ nouveau_fbcon_zfill(struct drm_device *dev, struct nouveau_fbdev *fbcon) ...@@ -251,9 +251,10 @@ nouveau_fbcon_zfill(struct drm_device *dev, struct nouveau_fbdev *fbcon)
} }
static int static int
nouveau_fbcon_create(struct nouveau_fbdev *fbcon, nouveau_fbcon_create(struct drm_fb_helper *helper,
struct drm_fb_helper_surface_size *sizes) struct drm_fb_helper_surface_size *sizes)
{ {
struct nouveau_fbdev *fbcon = (struct nouveau_fbdev *)helper;
struct drm_device *dev = fbcon->dev; struct drm_device *dev = fbcon->dev;
struct nouveau_drm *drm = nouveau_drm(dev); struct nouveau_drm *drm = nouveau_drm(dev);
struct nouveau_device *device = nv_device(drm->device); struct nouveau_device *device = nv_device(drm->device);
...@@ -388,23 +389,6 @@ nouveau_fbcon_create(struct nouveau_fbdev *fbcon, ...@@ -388,23 +389,6 @@ nouveau_fbcon_create(struct nouveau_fbdev *fbcon,
return ret; return ret;
} }
static int
nouveau_fbcon_find_or_create_single(struct drm_fb_helper *helper,
struct drm_fb_helper_surface_size *sizes)
{
struct nouveau_fbdev *fbcon = (struct nouveau_fbdev *)helper;
int new_fb = 0;
int ret;
if (!helper->fb) {
ret = nouveau_fbcon_create(fbcon, sizes);
if (ret)
return ret;
new_fb = 1;
}
return new_fb;
}
void void
nouveau_fbcon_output_poll_changed(struct drm_device *dev) nouveau_fbcon_output_poll_changed(struct drm_device *dev)
{ {
...@@ -450,7 +434,7 @@ void nouveau_fbcon_gpu_lockup(struct fb_info *info) ...@@ -450,7 +434,7 @@ void nouveau_fbcon_gpu_lockup(struct fb_info *info)
static struct drm_fb_helper_funcs nouveau_fbcon_helper_funcs = { static struct drm_fb_helper_funcs nouveau_fbcon_helper_funcs = {
.gamma_set = nouveau_fbcon_gamma_set, .gamma_set = nouveau_fbcon_gamma_set,
.gamma_get = nouveau_fbcon_gamma_get, .gamma_get = nouveau_fbcon_gamma_get,
.fb_probe = nouveau_fbcon_find_or_create_single, .fb_probe = nouveau_fbcon_create,
}; };
...@@ -491,6 +475,9 @@ nouveau_fbcon_init(struct drm_device *dev) ...@@ -491,6 +475,9 @@ nouveau_fbcon_init(struct drm_device *dev)
else else
preferred_bpp = 32; preferred_bpp = 32;
/* disable all the possible outputs/crtcs before entering KMS mode */
drm_helper_disable_unused_functions(dev);
drm_fb_helper_initial_config(&fbcon->helper, preferred_bpp); drm_fb_helper_initial_config(&fbcon->helper, preferred_bpp);
return 0; return 0;
} }
......
...@@ -187,9 +187,10 @@ static int radeonfb_create_pinned_object(struct radeon_fbdev *rfbdev, ...@@ -187,9 +187,10 @@ static int radeonfb_create_pinned_object(struct radeon_fbdev *rfbdev,
return ret; return ret;
} }
static int radeonfb_create(struct radeon_fbdev *rfbdev, static int radeonfb_create(struct drm_fb_helper *helper,
struct drm_fb_helper_surface_size *sizes) struct drm_fb_helper_surface_size *sizes)
{ {
struct radeon_fbdev *rfbdev = (struct radeon_fbdev *)helper;
struct radeon_device *rdev = rfbdev->rdev; struct radeon_device *rdev = rfbdev->rdev;
struct fb_info *info; struct fb_info *info;
struct drm_framebuffer *fb = NULL; struct drm_framebuffer *fb = NULL;
...@@ -300,22 +301,6 @@ static int radeonfb_create(struct radeon_fbdev *rfbdev, ...@@ -300,22 +301,6 @@ static int radeonfb_create(struct radeon_fbdev *rfbdev,
return ret; return ret;
} }
static int radeon_fb_find_or_create_single(struct drm_fb_helper *helper,
struct drm_fb_helper_surface_size *sizes)
{
struct radeon_fbdev *rfbdev = (struct radeon_fbdev *)helper;
int new_fb = 0;
int ret;
if (!helper->fb) {
ret = radeonfb_create(rfbdev, sizes);
if (ret)
return ret;
new_fb = 1;
}
return new_fb;
}
void radeon_fb_output_poll_changed(struct radeon_device *rdev) void radeon_fb_output_poll_changed(struct radeon_device *rdev)
{ {
drm_fb_helper_hotplug_event(&rdev->mode_info.rfbdev->helper); drm_fb_helper_hotplug_event(&rdev->mode_info.rfbdev->helper);
...@@ -349,7 +334,7 @@ static int radeon_fbdev_destroy(struct drm_device *dev, struct radeon_fbdev *rfb ...@@ -349,7 +334,7 @@ static int radeon_fbdev_destroy(struct drm_device *dev, struct radeon_fbdev *rfb
static struct drm_fb_helper_funcs radeon_fb_helper_funcs = { static struct drm_fb_helper_funcs radeon_fb_helper_funcs = {
.gamma_set = radeon_crtc_fb_gamma_set, .gamma_set = radeon_crtc_fb_gamma_set,
.gamma_get = radeon_crtc_fb_gamma_get, .gamma_get = radeon_crtc_fb_gamma_get,
.fb_probe = radeon_fb_find_or_create_single, .fb_probe = radeonfb_create,
}; };
int radeon_fbdev_init(struct radeon_device *rdev) int radeon_fbdev_init(struct radeon_device *rdev)
...@@ -379,6 +364,10 @@ int radeon_fbdev_init(struct radeon_device *rdev) ...@@ -379,6 +364,10 @@ int radeon_fbdev_init(struct radeon_device *rdev)
} }
drm_fb_helper_single_add_all_connectors(&rfbdev->helper); drm_fb_helper_single_add_all_connectors(&rfbdev->helper);
/* disable all the possible outputs/crtcs before entering KMS mode */
drm_helper_disable_unused_functions(rdev->ddev);
drm_fb_helper_initial_config(&rfbdev->helper, bpp_sel); drm_fb_helper_initial_config(&rfbdev->helper, bpp_sel);
return 0; return 0;
} }
......
...@@ -39,10 +39,6 @@ int tegra_drm_fb_init(struct drm_device *drm) ...@@ -39,10 +39,6 @@ int tegra_drm_fb_init(struct drm_device *drm)
if (IS_ERR(fbdev)) if (IS_ERR(fbdev))
return PTR_ERR(fbdev); return PTR_ERR(fbdev);
#ifndef CONFIG_FRAMEBUFFER_CONSOLE
drm_fbdev_cma_restore_mode(fbdev);
#endif
host1x->fbdev = fbdev; host1x->fbdev = fbdev;
return 0; return 0;
......
...@@ -476,9 +476,10 @@ udl_framebuffer_init(struct drm_device *dev, ...@@ -476,9 +476,10 @@ udl_framebuffer_init(struct drm_device *dev,
} }
static int udlfb_create(struct udl_fbdev *ufbdev, static int udlfb_create(struct drm_fb_helper *helper,
struct drm_fb_helper_surface_size *sizes) struct drm_fb_helper_surface_size *sizes)
{ {
struct udl_fbdev *ufbdev = (struct udl_fbdev *)helper;
struct drm_device *dev = ufbdev->helper.dev; struct drm_device *dev = ufbdev->helper.dev;
struct fb_info *info; struct fb_info *info;
struct device *device = &dev->usbdev->dev; struct device *device = &dev->usbdev->dev;
...@@ -556,27 +557,10 @@ static int udlfb_create(struct udl_fbdev *ufbdev, ...@@ -556,27 +557,10 @@ static int udlfb_create(struct udl_fbdev *ufbdev,
return ret; return ret;
} }
static int udl_fb_find_or_create_single(struct drm_fb_helper *helper,
struct drm_fb_helper_surface_size *sizes)
{
struct udl_fbdev *ufbdev = (struct udl_fbdev *)helper;
int new_fb = 0;
int ret;
if (!helper->fb) {
ret = udlfb_create(ufbdev, sizes);
if (ret)
return ret;
new_fb = 1;
}
return new_fb;
}
static struct drm_fb_helper_funcs udl_fb_helper_funcs = { static struct drm_fb_helper_funcs udl_fb_helper_funcs = {
.gamma_set = udl_crtc_fb_gamma_set, .gamma_set = udl_crtc_fb_gamma_set,
.gamma_get = udl_crtc_fb_gamma_get, .gamma_get = udl_crtc_fb_gamma_get,
.fb_probe = udl_fb_find_or_create_single, .fb_probe = udlfb_create,
}; };
static void udl_fbdev_destroy(struct drm_device *dev, static void udl_fbdev_destroy(struct drm_device *dev,
...@@ -619,6 +603,10 @@ int udl_fbdev_init(struct drm_device *dev) ...@@ -619,6 +603,10 @@ int udl_fbdev_init(struct drm_device *dev)
} }
drm_fb_helper_single_add_all_connectors(&ufbdev->helper); drm_fb_helper_single_add_all_connectors(&ufbdev->helper);
/* disable all the possible outputs/crtcs before entering KMS mode */
drm_helper_disable_unused_functions(dev);
drm_fb_helper_initial_config(&ufbdev->helper, bpp_sel); drm_fb_helper_initial_config(&ufbdev->helper, bpp_sel);
return 0; return 0;
} }
......
...@@ -296,25 +296,10 @@ static void omap_crtc_fb_gamma_get(struct drm_crtc *crtc, ...@@ -296,25 +296,10 @@ static void omap_crtc_fb_gamma_get(struct drm_crtc *crtc,
DBG("fbdev: get gamma"); DBG("fbdev: get gamma");
} }
static int omap_fbdev_probe(struct drm_fb_helper *helper,
struct drm_fb_helper_surface_size *sizes)
{
int new_fb = 0;
int ret;
if (!helper->fb) {
ret = omap_fbdev_create(helper, sizes);
if (ret)
return ret;
new_fb = 1;
}
return new_fb;
}
static struct drm_fb_helper_funcs omap_fb_helper_funcs = { static struct drm_fb_helper_funcs omap_fb_helper_funcs = {
.gamma_set = omap_crtc_fb_gamma_set, .gamma_set = omap_crtc_fb_gamma_set,
.gamma_get = omap_crtc_fb_gamma_get, .gamma_get = omap_crtc_fb_gamma_get,
.fb_probe = omap_fbdev_probe, .fb_probe = omap_fbdev_create,
}; };
static struct drm_fb_helper *get_fb(struct fb_info *fbi) static struct drm_fb_helper *get_fb(struct fb_info *fbi)
...@@ -369,6 +354,10 @@ struct drm_fb_helper *omap_fbdev_init(struct drm_device *dev) ...@@ -369,6 +354,10 @@ struct drm_fb_helper *omap_fbdev_init(struct drm_device *dev)
} }
drm_fb_helper_single_add_all_connectors(helper); drm_fb_helper_single_add_all_connectors(helper);
/* disable all the possible outputs/crtcs before entering KMS mode */
drm_helper_disable_unused_functions(dev);
drm_fb_helper_initial_config(helper, 32); drm_fb_helper_initial_config(helper, 32);
priv->fbdev = helper; priv->fbdev = helper;
......
...@@ -867,6 +867,7 @@ struct drm_prop_enum_list { ...@@ -867,6 +867,7 @@ struct drm_prop_enum_list {
extern void drm_modeset_lock_all(struct drm_device *dev); extern void drm_modeset_lock_all(struct drm_device *dev);
extern void drm_modeset_unlock_all(struct drm_device *dev); extern void drm_modeset_unlock_all(struct drm_device *dev);
extern void drm_warn_on_modeset_not_all_locked(struct drm_device *dev);
extern int drm_crtc_init(struct drm_device *dev, extern int drm_crtc_init(struct drm_device *dev,
struct drm_crtc *crtc, struct drm_crtc *crtc,
......
...@@ -48,6 +48,18 @@ struct drm_fb_helper_surface_size { ...@@ -48,6 +48,18 @@ struct drm_fb_helper_surface_size {
u32 surface_depth; u32 surface_depth;
}; };
/**
* struct drm_fb_helper_funcs - driver callbacks for the fbdev emulation library
* @gamma_set: - Set the given gamma lut register on the given crtc.
* @gamma_get: - Read the given gamma lut register on the given crtc, used to
* save the current lut when force-restoring the fbdev for e.g.
* kdbg.
* @fb_probe: - Driver callback to allocate and initialize the fbdev info
* structure. Futhermore it also needs to allocate the drm
* framebuffer used to back the fbdev.
*
* Driver callbacks used by the fbdev emulation helper library.
*/
struct drm_fb_helper_funcs { struct drm_fb_helper_funcs {
void (*gamma_set)(struct drm_crtc *crtc, u16 red, u16 green, void (*gamma_set)(struct drm_crtc *crtc, u16 red, u16 green,
u16 blue, int regno); u16 blue, int regno);
...@@ -65,9 +77,7 @@ struct drm_fb_helper_connector { ...@@ -65,9 +77,7 @@ struct drm_fb_helper_connector {
struct drm_fb_helper { struct drm_fb_helper {
struct drm_framebuffer *fb; struct drm_framebuffer *fb;
struct drm_framebuffer *saved_fb;
struct drm_device *dev; struct drm_device *dev;
struct drm_display_mode *mode;
int crtc_count; int crtc_count;
struct drm_fb_helper_crtc *crtc_info; struct drm_fb_helper_crtc *crtc_info;
int connector_count; int connector_count;
...@@ -82,9 +92,6 @@ struct drm_fb_helper { ...@@ -82,9 +92,6 @@ struct drm_fb_helper {
bool delayed_hotplug; bool delayed_hotplug;
}; };
int drm_fb_helper_single_fb_probe(struct drm_fb_helper *helper,
int preferred_bpp);
int drm_fb_helper_init(struct drm_device *dev, int drm_fb_helper_init(struct drm_device *dev,
struct drm_fb_helper *helper, int crtc_count, struct drm_fb_helper *helper, int crtc_count,
int max_conn); int max_conn);
...@@ -103,7 +110,6 @@ int drm_fb_helper_setcolreg(unsigned regno, ...@@ -103,7 +110,6 @@ int drm_fb_helper_setcolreg(unsigned regno,
struct fb_info *info); struct fb_info *info);
bool drm_fb_helper_restore_fbdev_mode(struct drm_fb_helper *fb_helper); bool drm_fb_helper_restore_fbdev_mode(struct drm_fb_helper *fb_helper);
void drm_fb_helper_restore(void);
void drm_fb_helper_fill_var(struct fb_info *info, struct drm_fb_helper *fb_helper, void drm_fb_helper_fill_var(struct fb_info *info, struct drm_fb_helper *fb_helper,
uint32_t fb_width, uint32_t fb_height); uint32_t fb_width, uint32_t fb_height);
void drm_fb_helper_fill_fix(struct fb_info *info, uint32_t pitch, void drm_fb_helper_fill_fix(struct fb_info *info, uint32_t pitch,
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment