2012-07-02 22:37:47 +08:00
|
|
|
/*
|
|
|
|
* drm kms/fb cma (contiguous memory allocator) helper functions
|
|
|
|
*
|
|
|
|
* Copyright (C) 2012 Analog Device Inc.
|
|
|
|
* Author: Lars-Peter Clausen <lars@metafoo.de>
|
|
|
|
*
|
|
|
|
* Based on udl_fbdev.c
|
|
|
|
* Copyright (C) 2012 Red Hat
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License
|
|
|
|
* as published by the Free Software Foundation; either version 2
|
|
|
|
* of the License, or (at your option) any later version.
|
|
|
|
* 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 <drm/drmP.h>
|
2018-07-04 00:03:50 +08:00
|
|
|
#include <drm/drm_client.h>
|
2012-07-02 22:37:47 +08:00
|
|
|
#include <drm/drm_fb_helper.h>
|
2017-08-13 21:31:45 +08:00
|
|
|
#include <drm/drm_framebuffer.h>
|
2012-07-02 22:37:47 +08:00
|
|
|
#include <drm/drm_gem_cma_helper.h>
|
2017-08-13 21:31:45 +08:00
|
|
|
#include <drm/drm_gem_framebuffer_helper.h>
|
2012-07-02 22:37:47 +08:00
|
|
|
#include <drm/drm_fb_cma_helper.h>
|
2017-11-15 22:19:41 +08:00
|
|
|
#include <drm/drm_print.h>
|
2012-07-02 22:37:47 +08:00
|
|
|
#include <linux/module.h>
|
|
|
|
|
|
|
|
struct drm_fbdev_cma {
|
|
|
|
struct drm_fb_helper fb_helper;
|
|
|
|
};
|
|
|
|
|
2016-04-28 23:18:35 +08:00
|
|
|
/**
|
|
|
|
* DOC: framebuffer cma helper functions
|
|
|
|
*
|
|
|
|
* Provides helper functions for creating a cma (contiguous memory allocator)
|
|
|
|
* backed framebuffer.
|
|
|
|
*
|
2017-09-24 20:26:25 +08:00
|
|
|
* drm_gem_fb_create() is used in the &drm_mode_config_funcs.fb_create
|
2016-05-12 00:09:18 +08:00
|
|
|
* callback function to create a cma backed framebuffer.
|
2016-04-28 23:18:35 +08:00
|
|
|
*
|
|
|
|
* An fbdev framebuffer backed by cma is also available by calling
|
2017-11-15 22:19:41 +08:00
|
|
|
* drm_fb_cma_fbdev_init(). drm_fb_cma_fbdev_fini() tears it down.
|
2016-04-28 23:18:35 +08:00
|
|
|
*/
|
|
|
|
|
2012-07-02 22:37:47 +08:00
|
|
|
static inline struct drm_fbdev_cma *to_fbdev_cma(struct drm_fb_helper *helper)
|
|
|
|
{
|
|
|
|
return container_of(helper, struct drm_fbdev_cma, fb_helper);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* drm_fb_cma_get_gem_obj() - Get CMA GEM object for framebuffer
|
|
|
|
* @fb: The framebuffer
|
|
|
|
* @plane: Which plane
|
|
|
|
*
|
|
|
|
* Return the CMA GEM object for given framebuffer.
|
|
|
|
*
|
|
|
|
* This function will usually be called from the CRTC callback functions.
|
|
|
|
*/
|
|
|
|
struct drm_gem_cma_object *drm_fb_cma_get_gem_obj(struct drm_framebuffer *fb,
|
2016-06-01 05:11:12 +08:00
|
|
|
unsigned int plane)
|
2012-07-02 22:37:47 +08:00
|
|
|
{
|
2017-08-13 21:31:45 +08:00
|
|
|
struct drm_gem_object *gem;
|
2012-07-02 22:37:47 +08:00
|
|
|
|
2017-08-13 21:31:45 +08:00
|
|
|
gem = drm_gem_fb_get_obj(fb, plane);
|
|
|
|
if (!gem)
|
2012-07-02 22:37:47 +08:00
|
|
|
return NULL;
|
|
|
|
|
2017-08-13 21:31:45 +08:00
|
|
|
return to_drm_gem_cma_obj(gem);
|
2012-07-02 22:37:47 +08:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(drm_fb_cma_get_gem_obj);
|
|
|
|
|
2017-04-14 18:13:32 +08:00
|
|
|
/**
|
|
|
|
* drm_fb_cma_get_gem_addr() - Get physical address for framebuffer
|
|
|
|
* @fb: The framebuffer
|
|
|
|
* @state: Which state of drm plane
|
|
|
|
* @plane: Which plane
|
|
|
|
* Return the CMA GEM address for given framebuffer.
|
|
|
|
*
|
|
|
|
* This function will usually be called from the PLANE callback functions.
|
|
|
|
*/
|
|
|
|
dma_addr_t drm_fb_cma_get_gem_addr(struct drm_framebuffer *fb,
|
|
|
|
struct drm_plane_state *state,
|
|
|
|
unsigned int plane)
|
|
|
|
{
|
2017-08-13 21:31:45 +08:00
|
|
|
struct drm_gem_cma_object *obj;
|
2017-04-14 18:13:32 +08:00
|
|
|
dma_addr_t paddr;
|
|
|
|
|
2017-08-13 21:31:45 +08:00
|
|
|
obj = drm_fb_cma_get_gem_obj(fb, plane);
|
|
|
|
if (!obj)
|
2017-04-14 18:13:32 +08:00
|
|
|
return 0;
|
|
|
|
|
2017-08-13 21:31:45 +08:00
|
|
|
paddr = obj->paddr + fb->offsets[plane];
|
2017-04-14 18:13:32 +08:00
|
|
|
paddr += fb->format->cpp[plane] * (state->src_x >> 16);
|
|
|
|
paddr += fb->pitches[plane] * (state->src_y >> 16);
|
|
|
|
|
|
|
|
return paddr;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(drm_fb_cma_get_gem_addr);
|
|
|
|
|
2017-11-15 22:19:41 +08:00
|
|
|
/**
|
|
|
|
* drm_fb_cma_fbdev_init() - Allocate and initialize fbdev emulation
|
|
|
|
* @dev: DRM device
|
|
|
|
* @preferred_bpp: Preferred bits per pixel for the device.
|
|
|
|
* @dev->mode_config.preferred_depth is used if this is zero.
|
|
|
|
* @max_conn_count: Maximum number of connectors.
|
|
|
|
* @dev->mode_config.num_connector is used if this is zero.
|
|
|
|
*
|
|
|
|
* Returns:
|
|
|
|
* Zero on success or negative error code on failure.
|
|
|
|
*/
|
|
|
|
int drm_fb_cma_fbdev_init(struct drm_device *dev, unsigned int preferred_bpp,
|
|
|
|
unsigned int max_conn_count)
|
|
|
|
{
|
2018-07-04 00:03:50 +08:00
|
|
|
struct drm_fbdev_cma *fbdev_cma;
|
|
|
|
|
|
|
|
/* dev->fb_helper will indirectly point to fbdev_cma after this call */
|
|
|
|
fbdev_cma = drm_fbdev_cma_init(dev, preferred_bpp, max_conn_count);
|
|
|
|
if (IS_ERR(fbdev_cma))
|
|
|
|
return PTR_ERR(fbdev_cma);
|
|
|
|
|
|
|
|
return 0;
|
2017-11-15 22:19:41 +08:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(drm_fb_cma_fbdev_init);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* drm_fb_cma_fbdev_fini() - Teardown fbdev emulation
|
|
|
|
* @dev: DRM device
|
|
|
|
*/
|
|
|
|
void drm_fb_cma_fbdev_fini(struct drm_device *dev)
|
|
|
|
{
|
2018-07-04 00:03:50 +08:00
|
|
|
if (dev->fb_helper)
|
|
|
|
drm_fbdev_cma_fini(to_fbdev_cma(dev->fb_helper));
|
2017-11-15 22:19:41 +08:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(drm_fb_cma_fbdev_fini);
|
|
|
|
|
2018-07-04 00:03:50 +08:00
|
|
|
static const struct drm_fb_helper_funcs drm_fb_cma_helper_funcs = {
|
|
|
|
.fb_probe = drm_fb_helper_generic_probe,
|
|
|
|
};
|
|
|
|
|
2012-07-02 22:37:47 +08:00
|
|
|
/**
|
2018-07-04 00:03:50 +08:00
|
|
|
* drm_fbdev_cma_init() - Allocate and initializes a drm_fbdev_cma struct
|
2012-07-02 22:37:47 +08:00
|
|
|
* @dev: DRM device
|
|
|
|
* @preferred_bpp: Preferred bits per pixel for the device
|
|
|
|
* @max_conn_count: Maximum number of connectors
|
|
|
|
*
|
|
|
|
* Returns a newly allocated drm_fbdev_cma struct or a ERR_PTR.
|
|
|
|
*/
|
2018-07-04 00:03:50 +08:00
|
|
|
struct drm_fbdev_cma *drm_fbdev_cma_init(struct drm_device *dev,
|
|
|
|
unsigned int preferred_bpp, unsigned int max_conn_count)
|
2012-07-02 22:37:47 +08:00
|
|
|
{
|
|
|
|
struct drm_fbdev_cma *fbdev_cma;
|
2018-07-04 00:03:50 +08:00
|
|
|
struct drm_fb_helper *fb_helper;
|
2012-07-02 22:37:47 +08:00
|
|
|
int ret;
|
|
|
|
|
|
|
|
fbdev_cma = kzalloc(sizeof(*fbdev_cma), GFP_KERNEL);
|
2018-07-04 00:03:50 +08:00
|
|
|
if (!fbdev_cma)
|
2012-07-02 22:37:47 +08:00
|
|
|
return ERR_PTR(-ENOMEM);
|
|
|
|
|
2018-07-04 00:03:50 +08:00
|
|
|
fb_helper = &fbdev_cma->fb_helper;
|
2014-06-27 23:19:24 +08:00
|
|
|
|
2018-07-04 00:03:50 +08:00
|
|
|
ret = drm_client_new(dev, &fb_helper->client, "fbdev", NULL);
|
|
|
|
if (ret)
|
2012-07-02 22:37:47 +08:00
|
|
|
goto err_free;
|
|
|
|
|
2018-07-04 00:03:50 +08:00
|
|
|
ret = drm_fb_helper_fbdev_setup(dev, fb_helper, &drm_fb_cma_helper_funcs,
|
|
|
|
preferred_bpp, max_conn_count);
|
|
|
|
if (ret)
|
|
|
|
goto err_client_put;
|
2012-07-02 22:37:47 +08:00
|
|
|
|
|
|
|
return fbdev_cma;
|
|
|
|
|
2018-07-04 00:03:50 +08:00
|
|
|
err_client_put:
|
|
|
|
drm_client_release(&fb_helper->client);
|
2012-07-02 22:37:47 +08:00
|
|
|
err_free:
|
|
|
|
kfree(fbdev_cma);
|
|
|
|
|
|
|
|
return ERR_PTR(ret);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(drm_fbdev_cma_init);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* drm_fbdev_cma_fini() - Free drm_fbdev_cma struct
|
|
|
|
* @fbdev_cma: The drm_fbdev_cma struct
|
|
|
|
*/
|
|
|
|
void drm_fbdev_cma_fini(struct drm_fbdev_cma *fbdev_cma)
|
|
|
|
{
|
2015-07-22 17:28:20 +08:00
|
|
|
drm_fb_helper_unregister_fbi(&fbdev_cma->fb_helper);
|
2018-07-04 00:03:50 +08:00
|
|
|
/* All resources have now been freed by drm_fbdev_fb_destroy() */
|
2012-07-02 22:37:47 +08:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(drm_fbdev_cma_fini);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* drm_fbdev_cma_restore_mode() - Restores initial framebuffer mode
|
|
|
|
* @fbdev_cma: The drm_fbdev_cma struct, may be NULL
|
|
|
|
*
|
2016-12-30 04:48:34 +08:00
|
|
|
* This function is usually called from the &drm_driver.lastclose callback.
|
2012-07-02 22:37:47 +08:00
|
|
|
*/
|
|
|
|
void drm_fbdev_cma_restore_mode(struct drm_fbdev_cma *fbdev_cma)
|
|
|
|
{
|
2014-05-31 00:29:48 +08:00
|
|
|
if (fbdev_cma)
|
|
|
|
drm_fb_helper_restore_fbdev_mode_unlocked(&fbdev_cma->fb_helper);
|
2012-07-02 22:37:47 +08:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(drm_fbdev_cma_restore_mode);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* drm_fbdev_cma_hotplug_event() - Poll for hotpulug events
|
|
|
|
* @fbdev_cma: The drm_fbdev_cma struct, may be NULL
|
|
|
|
*
|
2016-12-30 04:48:34 +08:00
|
|
|
* This function is usually called from the &drm_mode_config.output_poll_changed
|
2012-07-02 22:37:47 +08:00
|
|
|
* callback.
|
|
|
|
*/
|
|
|
|
void drm_fbdev_cma_hotplug_event(struct drm_fbdev_cma *fbdev_cma)
|
|
|
|
{
|
|
|
|
if (fbdev_cma)
|
|
|
|
drm_fb_helper_hotplug_event(&fbdev_cma->fb_helper);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(drm_fbdev_cma_hotplug_event);
|
2016-02-12 09:30:14 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* drm_fbdev_cma_set_suspend - wrapper around drm_fb_helper_set_suspend
|
|
|
|
* @fbdev_cma: The drm_fbdev_cma struct, may be NULL
|
|
|
|
* @state: desired state, zero to resume, non-zero to suspend
|
|
|
|
*
|
|
|
|
* Calls drm_fb_helper_set_suspend, which is a wrapper around
|
|
|
|
* fb_set_suspend implemented by fbdev core.
|
|
|
|
*/
|
2017-06-20 18:23:20 +08:00
|
|
|
void drm_fbdev_cma_set_suspend(struct drm_fbdev_cma *fbdev_cma, bool state)
|
2016-02-12 09:30:14 +08:00
|
|
|
{
|
|
|
|
if (fbdev_cma)
|
|
|
|
drm_fb_helper_set_suspend(&fbdev_cma->fb_helper, state);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(drm_fbdev_cma_set_suspend);
|
2017-01-23 02:11:09 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* drm_fbdev_cma_set_suspend_unlocked - wrapper around
|
|
|
|
* drm_fb_helper_set_suspend_unlocked
|
|
|
|
* @fbdev_cma: The drm_fbdev_cma struct, may be NULL
|
|
|
|
* @state: desired state, zero to resume, non-zero to suspend
|
|
|
|
*
|
|
|
|
* Calls drm_fb_helper_set_suspend, which is a wrapper around
|
|
|
|
* fb_set_suspend implemented by fbdev core.
|
|
|
|
*/
|
|
|
|
void drm_fbdev_cma_set_suspend_unlocked(struct drm_fbdev_cma *fbdev_cma,
|
2017-06-20 18:23:20 +08:00
|
|
|
bool state)
|
2017-01-23 02:11:09 +08:00
|
|
|
{
|
|
|
|
if (fbdev_cma)
|
|
|
|
drm_fb_helper_set_suspend_unlocked(&fbdev_cma->fb_helper,
|
|
|
|
state);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(drm_fbdev_cma_set_suspend_unlocked);
|