2 * Copyright (C) 2016 Noralf Trønnes
4 * This program is free software; you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License as published by
6 * the Free Software Foundation; either version 2 of the License, or
7 * (at your option) any later version.
10 #include <drm/drm_atomic.h>
11 #include <drm/drm_atomic_helper.h>
12 #include <drm/drm_crtc_helper.h>
13 #include <drm/drm_fb_helper.h>
14 #include <drm/drm_gem_framebuffer_helper.h>
15 #include <drm/tinydrm/tinydrm.h>
16 #include <linux/device.h>
17 #include <linux/dma-buf.h>
22 * This library provides driver helpers for very simple display hardware.
24 * It is based on &drm_simple_display_pipe coupled with a &drm_connector which
25 * has only one fixed &drm_display_mode. The framebuffers are backed by the
26 * cma helper and have support for framebuffer flushing (dirty).
27 * fbdev support is also included.
34 * The driver allocates &tinydrm_device, initializes it using
35 * devm_tinydrm_init(), sets up the pipeline using tinydrm_display_pipe_init()
36 * and registers the DRM device using devm_tinydrm_register().
40 * tinydrm_gem_cma_prime_import_sg_table - Produce a CMA GEM object from
41 * another driver's scatter/gather table of pinned pages
42 * @drm: DRM device to import into
43 * @attach: DMA-BUF attachment
44 * @sgt: Scatter/gather table of pinned pages
46 * This function imports a scatter/gather table exported via DMA-BUF by
47 * another driver using drm_gem_cma_prime_import_sg_table(). It sets the
48 * kernel virtual address on the CMA object. Drivers should use this as their
49 * &drm_driver->gem_prime_import_sg_table callback if they need the virtual
50 * address. tinydrm_gem_cma_free_object() should be used in combination with
54 * A pointer to a newly created GEM object or an ERR_PTR-encoded negative
55 * error code on failure.
57 struct drm_gem_object
*
58 tinydrm_gem_cma_prime_import_sg_table(struct drm_device
*drm
,
59 struct dma_buf_attachment
*attach
,
62 struct drm_gem_cma_object
*cma_obj
;
63 struct drm_gem_object
*obj
;
66 vaddr
= dma_buf_vmap(attach
->dmabuf
);
68 DRM_ERROR("Failed to vmap PRIME buffer\n");
69 return ERR_PTR(-ENOMEM
);
72 obj
= drm_gem_cma_prime_import_sg_table(drm
, attach
, sgt
);
74 dma_buf_vunmap(attach
->dmabuf
, vaddr
);
78 cma_obj
= to_drm_gem_cma_obj(obj
);
79 cma_obj
->vaddr
= vaddr
;
83 EXPORT_SYMBOL(tinydrm_gem_cma_prime_import_sg_table
);
86 * tinydrm_gem_cma_free_object - Free resources associated with a CMA GEM
88 * @gem_obj: GEM object to free
90 * This function frees the backing memory of the CMA GEM object, cleans up the
91 * GEM object state and frees the memory used to store the object itself using
92 * drm_gem_cma_free_object(). It also handles PRIME buffers which has the kernel
93 * virtual address set by tinydrm_gem_cma_prime_import_sg_table(). Drivers
94 * can use this as their &drm_driver->gem_free_object callback.
96 void tinydrm_gem_cma_free_object(struct drm_gem_object
*gem_obj
)
98 if (gem_obj
->import_attach
) {
99 struct drm_gem_cma_object
*cma_obj
;
101 cma_obj
= to_drm_gem_cma_obj(gem_obj
);
102 dma_buf_vunmap(gem_obj
->import_attach
->dmabuf
, cma_obj
->vaddr
);
103 cma_obj
->vaddr
= NULL
;
106 drm_gem_cma_free_object(gem_obj
);
108 EXPORT_SYMBOL_GPL(tinydrm_gem_cma_free_object
);
110 static struct drm_framebuffer
*
111 tinydrm_fb_create(struct drm_device
*drm
, struct drm_file
*file_priv
,
112 const struct drm_mode_fb_cmd2
*mode_cmd
)
114 struct tinydrm_device
*tdev
= drm
->dev_private
;
116 return drm_gem_fb_create_with_funcs(drm
, file_priv
, mode_cmd
,
120 static const struct drm_mode_config_funcs tinydrm_mode_config_funcs
= {
121 .fb_create
= tinydrm_fb_create
,
122 .atomic_check
= drm_atomic_helper_check
,
123 .atomic_commit
= drm_atomic_helper_commit
,
126 static int tinydrm_init(struct device
*parent
, struct tinydrm_device
*tdev
,
127 const struct drm_framebuffer_funcs
*fb_funcs
,
128 struct drm_driver
*driver
)
130 struct drm_device
*drm
;
132 mutex_init(&tdev
->dirty_lock
);
133 tdev
->fb_funcs
= fb_funcs
;
136 * We don't embed drm_device, because that prevent us from using
137 * devm_kzalloc() to allocate tinydrm_device in the driver since
138 * drm_dev_unref() frees the structure. The devm_ functions provide
139 * for easy error handling.
141 drm
= drm_dev_alloc(driver
, parent
);
146 drm
->dev_private
= tdev
;
147 drm_mode_config_init(drm
);
148 drm
->mode_config
.funcs
= &tinydrm_mode_config_funcs
;
153 static void tinydrm_fini(struct tinydrm_device
*tdev
)
155 drm_mode_config_cleanup(tdev
->drm
);
156 mutex_destroy(&tdev
->dirty_lock
);
157 tdev
->drm
->dev_private
= NULL
;
158 drm_dev_unref(tdev
->drm
);
161 static void devm_tinydrm_release(void *data
)
167 * devm_tinydrm_init - Initialize tinydrm device
168 * @parent: Parent device object
169 * @tdev: tinydrm device
170 * @fb_funcs: Framebuffer functions
171 * @driver: DRM driver
173 * This function initializes @tdev, the underlying DRM device and it's
174 * mode_config. Resources will be automatically freed on driver detach (devres)
175 * using drm_mode_config_cleanup() and drm_dev_unref().
178 * Zero on success, negative error code on failure.
180 int devm_tinydrm_init(struct device
*parent
, struct tinydrm_device
*tdev
,
181 const struct drm_framebuffer_funcs
*fb_funcs
,
182 struct drm_driver
*driver
)
186 ret
= tinydrm_init(parent
, tdev
, fb_funcs
, driver
);
190 ret
= devm_add_action(parent
, devm_tinydrm_release
, tdev
);
196 EXPORT_SYMBOL(devm_tinydrm_init
);
198 static int tinydrm_register(struct tinydrm_device
*tdev
)
200 struct drm_device
*drm
= tdev
->drm
;
203 ret
= drm_dev_register(tdev
->drm
, 0);
207 ret
= drm_fb_cma_fbdev_init_with_funcs(drm
, 0, 0, tdev
->fb_funcs
);
209 DRM_ERROR("Failed to initialize fbdev: %d\n", ret
);
214 static void tinydrm_unregister(struct tinydrm_device
*tdev
)
216 drm_atomic_helper_shutdown(tdev
->drm
);
217 drm_fb_cma_fbdev_fini(tdev
->drm
);
218 drm_dev_unregister(tdev
->drm
);
221 static void devm_tinydrm_register_release(void *data
)
223 tinydrm_unregister(data
);
227 * devm_tinydrm_register - Register tinydrm device
228 * @tdev: tinydrm device
230 * This function registers the underlying DRM device and fbdev.
231 * These resources will be automatically unregistered on driver detach (devres)
232 * and the display pipeline will be disabled.
235 * Zero on success, negative error code on failure.
237 int devm_tinydrm_register(struct tinydrm_device
*tdev
)
239 struct device
*dev
= tdev
->drm
->dev
;
242 ret
= tinydrm_register(tdev
);
246 ret
= devm_add_action(dev
, devm_tinydrm_register_release
, tdev
);
248 tinydrm_unregister(tdev
);
252 EXPORT_SYMBOL(devm_tinydrm_register
);
255 * tinydrm_shutdown - Shutdown tinydrm
256 * @tdev: tinydrm device
258 * This function makes sure that the display pipeline is disabled.
259 * Used by drivers in their shutdown callback to turn off the display
260 * on machine shutdown and reboot.
262 void tinydrm_shutdown(struct tinydrm_device
*tdev
)
264 drm_atomic_helper_shutdown(tdev
->drm
);
266 EXPORT_SYMBOL(tinydrm_shutdown
);
268 MODULE_LICENSE("GPL");