blob: b360e6251836d990e382f0f7451788054336a0d5 [file] [log] [blame]
Mark Yao2048e322014-08-22 18:36:26 +08001/*
2 * Copyright (C) Fuzhou Rockchip Electronics Co.Ltd
3 * Author:Mark Yao <mark.yao@rock-chips.com>
4 *
5 * based on exynos_drm_drv.c
6 *
7 * This software is licensed under the terms of the GNU General Public
8 * License version 2, as published by the Free Software Foundation, and
9 * may be copied, distributed, and modified under those terms.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
15 */
16
Mark Yao2048e322014-08-22 18:36:26 +080017#include <drm/drmP.h>
18#include <drm/drm_crtc_helper.h>
19#include <drm/drm_fb_helper.h>
Daniel Vetter80f67cd2016-05-30 19:53:12 +020020#include <drm/drm_gem_cma_helper.h>
Russell King97ac0e42016-10-19 11:28:27 +010021#include <drm/drm_of.h>
Mark Yao2048e322014-08-22 18:36:26 +080022#include <linux/dma-mapping.h>
Shunqian Zheng1aa5ca62016-06-24 10:13:32 +080023#include <linux/dma-iommu.h>
Mark Yao2048e322014-08-22 18:36:26 +080024#include <linux/pm_runtime.h>
Paul Gortmaker00fe6142015-05-01 20:02:30 -040025#include <linux/module.h>
Mark Yao2048e322014-08-22 18:36:26 +080026#include <linux/of_graph.h>
27#include <linux/component.h>
Tomeu Vizoso5a587382016-06-06 16:53:32 +020028#include <linux/console.h>
Shunqian Zheng1aa5ca62016-06-24 10:13:32 +080029#include <linux/iommu.h>
Mark Yao2048e322014-08-22 18:36:26 +080030
31#include "rockchip_drm_drv.h"
32#include "rockchip_drm_fb.h"
33#include "rockchip_drm_fbdev.h"
34#include "rockchip_drm_gem.h"
35
36#define DRIVER_NAME "rockchip"
37#define DRIVER_DESC "RockChip Soc DRM"
38#define DRIVER_DATE "20140818"
39#define DRIVER_MAJOR 1
40#define DRIVER_MINOR 0
41
Mark Yao2d90d472016-04-19 10:13:27 +080042static bool is_support_iommu = true;
Tomeu Vizosof7069742016-06-10 13:14:13 +020043static struct drm_driver rockchip_drm_driver;
Mark Yao2d90d472016-04-19 10:13:27 +080044
Mark Yao2048e322014-08-22 18:36:26 +080045/*
46 * Attach a (component) device to the shared drm dma mapping from master drm
47 * device. This is used by the VOPs to map GEM buffers to a common DMA
48 * mapping.
49 */
50int rockchip_drm_dma_attach_device(struct drm_device *drm_dev,
51 struct device *dev)
52{
Shunqian Zheng1aa5ca62016-06-24 10:13:32 +080053 struct rockchip_drm_private *private = drm_dev->dev_private;
Mark Yao2048e322014-08-22 18:36:26 +080054 int ret;
55
Mark Yao2d90d472016-04-19 10:13:27 +080056 if (!is_support_iommu)
57 return 0;
58
Shunqian Zheng1aa5ca62016-06-24 10:13:32 +080059 ret = iommu_attach_device(private->domain, dev);
60 if (ret) {
61 dev_err(dev, "Failed to attach iommu device\n");
Mark Yao2048e322014-08-22 18:36:26 +080062 return ret;
Shunqian Zheng1aa5ca62016-06-24 10:13:32 +080063 }
Mark Yao2048e322014-08-22 18:36:26 +080064
Shunqian Zheng1aa5ca62016-06-24 10:13:32 +080065 return 0;
Mark Yao2048e322014-08-22 18:36:26 +080066}
Mark Yao2048e322014-08-22 18:36:26 +080067
68void rockchip_drm_dma_detach_device(struct drm_device *drm_dev,
69 struct device *dev)
70{
Shunqian Zheng1aa5ca62016-06-24 10:13:32 +080071 struct rockchip_drm_private *private = drm_dev->dev_private;
72 struct iommu_domain *domain = private->domain;
73
Mark Yao2d90d472016-04-19 10:13:27 +080074 if (!is_support_iommu)
75 return;
76
Shunqian Zheng1aa5ca62016-06-24 10:13:32 +080077 iommu_detach_device(domain, dev);
Mark Yao2048e322014-08-22 18:36:26 +080078}
Mark Yao2048e322014-08-22 18:36:26 +080079
Mark Yaob5f7b752015-11-23 15:21:08 +080080int rockchip_register_crtc_funcs(struct drm_crtc *crtc,
81 const struct rockchip_crtc_funcs *crtc_funcs)
Mark Yao2048e322014-08-22 18:36:26 +080082{
Mark Yaob5f7b752015-11-23 15:21:08 +080083 int pipe = drm_crtc_index(crtc);
84 struct rockchip_drm_private *priv = crtc->dev->dev_private;
Mark Yao2048e322014-08-22 18:36:26 +080085
Dan Carpenter15da7802016-07-13 13:15:04 +030086 if (pipe >= ROCKCHIP_MAX_CRTC)
Mark Yao2048e322014-08-22 18:36:26 +080087 return -EINVAL;
88
89 priv->crtc_funcs[pipe] = crtc_funcs;
90
91 return 0;
92}
Mark Yao2048e322014-08-22 18:36:26 +080093
Mark Yaob5f7b752015-11-23 15:21:08 +080094void rockchip_unregister_crtc_funcs(struct drm_crtc *crtc)
Mark Yao2048e322014-08-22 18:36:26 +080095{
Mark Yaob5f7b752015-11-23 15:21:08 +080096 int pipe = drm_crtc_index(crtc);
97 struct rockchip_drm_private *priv = crtc->dev->dev_private;
Mark Yao2048e322014-08-22 18:36:26 +080098
Dan Carpenter15da7802016-07-13 13:15:04 +030099 if (pipe >= ROCKCHIP_MAX_CRTC)
Mark Yao2048e322014-08-22 18:36:26 +0800100 return;
101
102 priv->crtc_funcs[pipe] = NULL;
103}
Mark Yao2048e322014-08-22 18:36:26 +0800104
Thierry Reding88e72712015-09-24 18:35:31 +0200105static int rockchip_drm_crtc_enable_vblank(struct drm_device *dev,
106 unsigned int pipe)
Mark Yao2048e322014-08-22 18:36:26 +0800107{
108 struct rockchip_drm_private *priv = dev->dev_private;
Shawn Guo48df6332016-12-29 20:41:30 +0800109 struct drm_crtc *crtc = drm_crtc_from_index(dev, pipe);
Mark Yao2048e322014-08-22 18:36:26 +0800110
111 if (crtc && priv->crtc_funcs[pipe] &&
112 priv->crtc_funcs[pipe]->enable_vblank)
113 return priv->crtc_funcs[pipe]->enable_vblank(crtc);
114
115 return 0;
116}
117
Thierry Reding88e72712015-09-24 18:35:31 +0200118static void rockchip_drm_crtc_disable_vblank(struct drm_device *dev,
119 unsigned int pipe)
Mark Yao2048e322014-08-22 18:36:26 +0800120{
121 struct rockchip_drm_private *priv = dev->dev_private;
Shawn Guo48df6332016-12-29 20:41:30 +0800122 struct drm_crtc *crtc = drm_crtc_from_index(dev, pipe);
Mark Yao2048e322014-08-22 18:36:26 +0800123
124 if (crtc && priv->crtc_funcs[pipe] &&
125 priv->crtc_funcs[pipe]->enable_vblank)
126 priv->crtc_funcs[pipe]->disable_vblank(crtc);
127}
128
Shunqian Zheng1aa5ca62016-06-24 10:13:32 +0800129static int rockchip_drm_init_iommu(struct drm_device *drm_dev)
130{
131 struct rockchip_drm_private *private = drm_dev->dev_private;
132 struct iommu_domain_geometry *geometry;
133 u64 start, end;
134
135 if (!is_support_iommu)
136 return 0;
137
138 private->domain = iommu_domain_alloc(&platform_bus_type);
139 if (!private->domain)
140 return -ENOMEM;
141
142 geometry = &private->domain->geometry;
143 start = geometry->aperture_start;
144 end = geometry->aperture_end;
145
146 DRM_DEBUG("IOMMU context initialized (aperture: %#llx-%#llx)\n",
147 start, end);
148 drm_mm_init(&private->mm, start, end - start + 1);
149 mutex_init(&private->mm_lock);
150
151 return 0;
152}
153
154static void rockchip_iommu_cleanup(struct drm_device *drm_dev)
155{
156 struct rockchip_drm_private *private = drm_dev->dev_private;
157
158 if (!is_support_iommu)
159 return;
160
161 drm_mm_takedown(&private->mm);
162 iommu_domain_free(private->domain);
163}
164
Tomeu Vizosof7069742016-06-10 13:14:13 +0200165static int rockchip_drm_bind(struct device *dev)
Mark Yao2048e322014-08-22 18:36:26 +0800166{
Tomeu Vizosof7069742016-06-10 13:14:13 +0200167 struct drm_device *drm_dev;
Mark Yao2048e322014-08-22 18:36:26 +0800168 struct rockchip_drm_private *private;
Mark Yao2048e322014-08-22 18:36:26 +0800169 int ret;
170
Tomeu Vizosof7069742016-06-10 13:14:13 +0200171 drm_dev = drm_dev_alloc(&rockchip_drm_driver, dev);
Tom Gundersen0f288602016-09-21 16:59:19 +0200172 if (IS_ERR(drm_dev))
173 return PTR_ERR(drm_dev);
Mark Yao2048e322014-08-22 18:36:26 +0800174
Tomeu Vizosof7069742016-06-10 13:14:13 +0200175 dev_set_drvdata(dev, drm_dev);
176
177 private = devm_kzalloc(drm_dev->dev, sizeof(*private), GFP_KERNEL);
178 if (!private) {
179 ret = -ENOMEM;
Tomasz Figa9127f992016-06-21 13:27:34 +0900180 goto err_free;
Tomeu Vizosof7069742016-06-10 13:14:13 +0200181 }
182
Mark Yao2048e322014-08-22 18:36:26 +0800183 drm_dev->dev_private = private;
184
Yakir Yang5182c1a2016-07-24 14:57:44 +0800185 INIT_LIST_HEAD(&private->psr_list);
Sean Paul18d8d4d2016-08-16 16:11:28 -0700186 spin_lock_init(&private->psr_list_lock);
Yakir Yang5182c1a2016-07-24 14:57:44 +0800187
Mark Yao2048e322014-08-22 18:36:26 +0800188 drm_mode_config_init(drm_dev);
189
190 rockchip_drm_mode_config_init(drm_dev);
191
Shunqian Zheng1aa5ca62016-06-24 10:13:32 +0800192 ret = rockchip_drm_init_iommu(drm_dev);
193 if (ret)
Mark Yao2048e322014-08-22 18:36:26 +0800194 goto err_config_cleanup;
Mark Yao2048e322014-08-22 18:36:26 +0800195
Mark Yao2048e322014-08-22 18:36:26 +0800196 /* Try to bind all sub drivers. */
197 ret = component_bind_all(dev, drm_dev);
198 if (ret)
Shunqian Zheng1aa5ca62016-06-24 10:13:32 +0800199 goto err_iommu_cleanup;
Mark Yao2048e322014-08-22 18:36:26 +0800200
201 /* init kms poll for handling hpd */
202 drm_kms_helper_poll_init(drm_dev);
203
204 /*
205 * enable drm irq mode.
206 * - with irq_enabled = true, we can use the vblank feature.
207 */
208 drm_dev->irq_enabled = true;
209
210 ret = drm_vblank_init(drm_dev, ROCKCHIP_MAX_CRTC);
211 if (ret)
212 goto err_kms_helper_poll_fini;
213
Mark Yao63ebb9f2015-11-30 18:22:42 +0800214 drm_mode_config_reset(drm_dev);
215
Mark Yao2048e322014-08-22 18:36:26 +0800216 ret = rockchip_drm_fbdev_init(drm_dev);
217 if (ret)
218 goto err_vblank_cleanup;
219
Tomasz Figa9127f992016-06-21 13:27:34 +0900220 ret = drm_dev_register(drm_dev, 0);
221 if (ret)
222 goto err_fbdev_fini;
223
Mark Yao2048e322014-08-22 18:36:26 +0800224 return 0;
Tomasz Figa9127f992016-06-21 13:27:34 +0900225err_fbdev_fini:
226 rockchip_drm_fbdev_fini(drm_dev);
Mark Yao2048e322014-08-22 18:36:26 +0800227err_vblank_cleanup:
228 drm_vblank_cleanup(drm_dev);
229err_kms_helper_poll_fini:
230 drm_kms_helper_poll_fini(drm_dev);
231 component_unbind_all(dev, drm_dev);
Shunqian Zheng1aa5ca62016-06-24 10:13:32 +0800232err_iommu_cleanup:
233 rockchip_iommu_cleanup(drm_dev);
Mark Yao2048e322014-08-22 18:36:26 +0800234err_config_cleanup:
235 drm_mode_config_cleanup(drm_dev);
236 drm_dev->dev_private = NULL;
Tomeu Vizosof7069742016-06-10 13:14:13 +0200237err_free:
238 drm_dev_unref(drm_dev);
Mark Yao2048e322014-08-22 18:36:26 +0800239 return ret;
240}
241
Tomeu Vizosof7069742016-06-10 13:14:13 +0200242static void rockchip_drm_unbind(struct device *dev)
Mark Yao2048e322014-08-22 18:36:26 +0800243{
Tomeu Vizosof7069742016-06-10 13:14:13 +0200244 struct drm_device *drm_dev = dev_get_drvdata(dev);
Mark Yao2048e322014-08-22 18:36:26 +0800245
246 rockchip_drm_fbdev_fini(drm_dev);
247 drm_vblank_cleanup(drm_dev);
248 drm_kms_helper_poll_fini(drm_dev);
249 component_unbind_all(dev, drm_dev);
Shunqian Zheng1aa5ca62016-06-24 10:13:32 +0800250 rockchip_iommu_cleanup(drm_dev);
Mark Yao2048e322014-08-22 18:36:26 +0800251 drm_mode_config_cleanup(drm_dev);
252 drm_dev->dev_private = NULL;
Tomeu Vizosof7069742016-06-10 13:14:13 +0200253 drm_dev_unregister(drm_dev);
254 drm_dev_unref(drm_dev);
255 dev_set_drvdata(dev, NULL);
Mark Yao2048e322014-08-22 18:36:26 +0800256}
257
John Keeping8ff490a2016-05-10 17:03:56 +0100258static void rockchip_drm_lastclose(struct drm_device *dev)
Mark Yao2048e322014-08-22 18:36:26 +0800259{
260 struct rockchip_drm_private *priv = dev->dev_private;
261
262 drm_fb_helper_restore_fbdev_mode_unlocked(&priv->fbdev_helper);
263}
264
265static const struct file_operations rockchip_drm_driver_fops = {
266 .owner = THIS_MODULE,
267 .open = drm_open,
268 .mmap = rockchip_gem_mmap,
269 .poll = drm_poll,
270 .read = drm_read,
271 .unlocked_ioctl = drm_ioctl,
Mark Yao2048e322014-08-22 18:36:26 +0800272 .compat_ioctl = drm_compat_ioctl,
Mark Yao2048e322014-08-22 18:36:26 +0800273 .release = drm_release,
274};
275
Mark Yao2048e322014-08-22 18:36:26 +0800276static struct drm_driver rockchip_drm_driver = {
Mark Yao63ebb9f2015-11-30 18:22:42 +0800277 .driver_features = DRIVER_MODESET | DRIVER_GEM |
278 DRIVER_PRIME | DRIVER_ATOMIC,
Mark Yao2048e322014-08-22 18:36:26 +0800279 .lastclose = rockchip_drm_lastclose,
Ville Syrjäläb44f8402015-09-30 16:46:48 +0300280 .get_vblank_counter = drm_vblank_no_hw_counter,
Mark Yao2048e322014-08-22 18:36:26 +0800281 .enable_vblank = rockchip_drm_crtc_enable_vblank,
282 .disable_vblank = rockchip_drm_crtc_disable_vblank,
Daniel Vetter80f67cd2016-05-30 19:53:12 +0200283 .gem_vm_ops = &drm_gem_cma_vm_ops,
Daniel Vetterc2466ac2016-05-30 19:53:03 +0200284 .gem_free_object_unlocked = rockchip_gem_free_object,
Mark Yao2048e322014-08-22 18:36:26 +0800285 .dumb_create = rockchip_gem_dumb_create,
286 .dumb_map_offset = rockchip_gem_dumb_map_offset,
287 .dumb_destroy = drm_gem_dumb_destroy,
288 .prime_handle_to_fd = drm_gem_prime_handle_to_fd,
289 .prime_fd_to_handle = drm_gem_prime_fd_to_handle,
290 .gem_prime_import = drm_gem_prime_import,
291 .gem_prime_export = drm_gem_prime_export,
292 .gem_prime_get_sg_table = rockchip_gem_prime_get_sg_table,
293 .gem_prime_vmap = rockchip_gem_prime_vmap,
294 .gem_prime_vunmap = rockchip_gem_prime_vunmap,
295 .gem_prime_mmap = rockchip_gem_mmap_buf,
296 .fops = &rockchip_drm_driver_fops,
297 .name = DRIVER_NAME,
298 .desc = DRIVER_DESC,
299 .date = DRIVER_DATE,
300 .major = DRIVER_MAJOR,
301 .minor = DRIVER_MINOR,
302};
303
304#ifdef CONFIG_PM_SLEEP
Baoyou Xie623981772016-09-25 15:43:08 +0800305static void rockchip_drm_fb_suspend(struct drm_device *drm)
Tomeu Vizoso5a587382016-06-06 16:53:32 +0200306{
307 struct rockchip_drm_private *priv = drm->dev_private;
308
309 console_lock();
310 drm_fb_helper_set_suspend(&priv->fbdev_helper, 1);
311 console_unlock();
312}
313
Baoyou Xie623981772016-09-25 15:43:08 +0800314static void rockchip_drm_fb_resume(struct drm_device *drm)
Tomeu Vizoso5a587382016-06-06 16:53:32 +0200315{
316 struct rockchip_drm_private *priv = drm->dev_private;
317
318 console_lock();
319 drm_fb_helper_set_suspend(&priv->fbdev_helper, 0);
320 console_unlock();
321}
322
Mark Yao2048e322014-08-22 18:36:26 +0800323static int rockchip_drm_sys_suspend(struct device *dev)
324{
325 struct drm_device *drm = dev_get_drvdata(dev);
Tomeu Vizoso5a587382016-06-06 16:53:32 +0200326 struct rockchip_drm_private *priv = drm->dev_private;
Mark Yao2048e322014-08-22 18:36:26 +0800327
Tomeu Vizoso5a587382016-06-06 16:53:32 +0200328 drm_kms_helper_poll_disable(drm);
329 rockchip_drm_fb_suspend(drm);
Mark Yao2048e322014-08-22 18:36:26 +0800330
Tomeu Vizoso5a587382016-06-06 16:53:32 +0200331 priv->state = drm_atomic_helper_suspend(drm);
332 if (IS_ERR(priv->state)) {
333 rockchip_drm_fb_resume(drm);
334 drm_kms_helper_poll_enable(drm);
335 return PTR_ERR(priv->state);
Mark Yao2048e322014-08-22 18:36:26 +0800336 }
Mark Yao2048e322014-08-22 18:36:26 +0800337
338 return 0;
339}
340
341static int rockchip_drm_sys_resume(struct device *dev)
342{
343 struct drm_device *drm = dev_get_drvdata(dev);
Tomeu Vizoso5a587382016-06-06 16:53:32 +0200344 struct rockchip_drm_private *priv = drm->dev_private;
Mark Yao2048e322014-08-22 18:36:26 +0800345
Tomeu Vizoso5a587382016-06-06 16:53:32 +0200346 drm_atomic_helper_resume(drm, priv->state);
347 rockchip_drm_fb_resume(drm);
348 drm_kms_helper_poll_enable(drm);
Mark Yao2048e322014-08-22 18:36:26 +0800349
350 return 0;
351}
352#endif
353
354static const struct dev_pm_ops rockchip_drm_pm_ops = {
355 SET_SYSTEM_SLEEP_PM_OPS(rockchip_drm_sys_suspend,
356 rockchip_drm_sys_resume)
357};
358
Mark Yao2048e322014-08-22 18:36:26 +0800359static int compare_of(struct device *dev, void *data)
360{
361 struct device_node *np = data;
362
363 return dev->of_node == np;
364}
365
Mark Yao5bad7d22015-11-10 16:47:19 +0800366static void rockchip_add_endpoints(struct device *dev,
367 struct component_match **match,
368 struct device_node *port)
369{
370 struct device_node *ep, *remote;
371
372 for_each_child_of_node(port, ep) {
373 remote = of_graph_get_remote_port_parent(ep);
374 if (!remote || !of_device_is_available(remote)) {
375 of_node_put(remote);
376 continue;
377 } else if (!of_device_is_available(remote->parent)) {
378 dev_warn(dev, "parent device of %s is not available\n",
379 remote->full_name);
380 of_node_put(remote);
381 continue;
382 }
383
Russell King97ac0e42016-10-19 11:28:27 +0100384 drm_of_component_match_add(dev, match, compare_of, remote);
Mark Yao5bad7d22015-11-10 16:47:19 +0800385 of_node_put(remote);
386 }
387}
388
Mark Yao2048e322014-08-22 18:36:26 +0800389static const struct component_master_ops rockchip_drm_ops = {
390 .bind = rockchip_drm_bind,
391 .unbind = rockchip_drm_unbind,
392};
393
394static int rockchip_drm_platform_probe(struct platform_device *pdev)
395{
Mark Yao5bad7d22015-11-10 16:47:19 +0800396 struct device *dev = &pdev->dev;
397 struct component_match *match = NULL;
398 struct device_node *np = dev->of_node;
399 struct device_node *port;
400 int i;
Mark Yao2048e322014-08-22 18:36:26 +0800401
Mark Yao5bad7d22015-11-10 16:47:19 +0800402 if (!np)
Mark Yao2048e322014-08-22 18:36:26 +0800403 return -ENODEV;
Mark Yao5bad7d22015-11-10 16:47:19 +0800404 /*
405 * Bind the crtc ports first, so that
406 * drm_of_find_possible_crtcs called from encoder .bind callbacks
407 * works as expected.
408 */
409 for (i = 0;; i++) {
Mark Yao2d90d472016-04-19 10:13:27 +0800410 struct device_node *iommu;
411
Mark Yao5bad7d22015-11-10 16:47:19 +0800412 port = of_parse_phandle(np, "ports", i);
413 if (!port)
414 break;
Mark Yao2048e322014-08-22 18:36:26 +0800415
Mark Yao5bad7d22015-11-10 16:47:19 +0800416 if (!of_device_is_available(port->parent)) {
417 of_node_put(port);
418 continue;
419 }
420
Mark Yao2d90d472016-04-19 10:13:27 +0800421 iommu = of_parse_phandle(port->parent, "iommus", 0);
422 if (!iommu || !of_device_is_available(iommu->parent)) {
423 dev_dbg(dev, "no iommu attached for %s, using non-iommu buffers\n",
424 port->parent->full_name);
425 /*
426 * if there is a crtc not support iommu, force set all
427 * crtc use non-iommu buffer.
428 */
429 is_support_iommu = false;
430 }
431
Peter Chen6d5fa282016-07-05 10:04:48 +0800432 of_node_put(iommu);
Russell King97ac0e42016-10-19 11:28:27 +0100433 drm_of_component_match_add(dev, &match, compare_of,
434 port->parent);
Mark Yao5bad7d22015-11-10 16:47:19 +0800435 of_node_put(port);
436 }
437
438 if (i == 0) {
439 dev_err(dev, "missing 'ports' property\n");
440 return -ENODEV;
441 }
442
443 if (!match) {
444 dev_err(dev, "No available vop found for display-subsystem.\n");
445 return -ENODEV;
446 }
447 /*
448 * For each bound crtc, bind the encoders attached to its
449 * remote endpoint.
450 */
451 for (i = 0;; i++) {
452 port = of_parse_phandle(np, "ports", i);
453 if (!port)
454 break;
455
456 if (!of_device_is_available(port->parent)) {
457 of_node_put(port);
458 continue;
459 }
460
461 rockchip_add_endpoints(dev, &match, port);
462 of_node_put(port);
463 }
464
465 return component_master_add_with_match(dev, &rockchip_drm_ops, match);
Mark Yao2048e322014-08-22 18:36:26 +0800466}
467
468static int rockchip_drm_platform_remove(struct platform_device *pdev)
469{
470 component_master_del(&pdev->dev, &rockchip_drm_ops);
471
472 return 0;
473}
474
475static const struct of_device_id rockchip_drm_dt_ids[] = {
476 { .compatible = "rockchip,display-subsystem", },
477 { /* sentinel */ },
478};
479MODULE_DEVICE_TABLE(of, rockchip_drm_dt_ids);
480
481static struct platform_driver rockchip_drm_platform_driver = {
482 .probe = rockchip_drm_platform_probe,
483 .remove = rockchip_drm_platform_remove,
484 .driver = {
Mark Yao2048e322014-08-22 18:36:26 +0800485 .name = "rockchip-drm",
486 .of_match_table = rockchip_drm_dt_ids,
487 .pm = &rockchip_drm_pm_ops,
488 },
489};
490
491module_platform_driver(rockchip_drm_platform_driver);
492
493MODULE_AUTHOR("Mark Yao <mark.yao@rock-chips.com>");
494MODULE_DESCRIPTION("ROCKCHIP DRM Driver");
495MODULE_LICENSE("GPL v2");