exynos_drm_core.c 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162
  1. /* exynos_drm_core.c
  2. *
  3. * Copyright (c) 2011 Samsung Electronics Co., Ltd.
  4. * Author:
  5. * Inki Dae <inki.dae@samsung.com>
  6. * Joonyoung Shim <jy0922.shim@samsung.com>
  7. * Seung-Woo Kim <sw0312.kim@samsung.com>
  8. *
  9. * This program is free software; you can redistribute it and/or modify it
  10. * under the terms of the GNU General Public License as published by the
  11. * Free Software Foundation; either version 2 of the License, or (at your
  12. * option) any later version.
  13. */
  14. #include <drm/drmP.h>
  15. #include "exynos_drm_drv.h"
  16. #include "exynos_drm_crtc.h"
  17. #include "exynos_drm_encoder.h"
  18. #include "exynos_drm_fbdev.h"
  19. static LIST_HEAD(exynos_drm_subdrv_list);
  20. int exynos_drm_create_enc_conn(struct drm_device *dev,
  21. struct exynos_drm_display *display)
  22. {
  23. struct drm_encoder *encoder;
  24. int ret;
  25. unsigned long possible_crtcs = 0;
  26. ret = exynos_drm_crtc_get_pipe_from_type(dev, display->type);
  27. if (ret < 0)
  28. return ret;
  29. possible_crtcs |= 1 << ret;
  30. /* create and initialize a encoder for this sub driver. */
  31. encoder = exynos_drm_encoder_create(dev, display, possible_crtcs);
  32. if (!encoder) {
  33. DRM_ERROR("failed to create encoder\n");
  34. return -EFAULT;
  35. }
  36. display->encoder = encoder;
  37. ret = display->ops->create_connector(display, encoder);
  38. if (ret) {
  39. DRM_ERROR("failed to create connector ret = %d\n", ret);
  40. goto err_destroy_encoder;
  41. }
  42. return 0;
  43. err_destroy_encoder:
  44. encoder->funcs->destroy(encoder);
  45. return ret;
  46. }
  47. int exynos_drm_subdrv_register(struct exynos_drm_subdrv *subdrv)
  48. {
  49. if (!subdrv)
  50. return -EINVAL;
  51. list_add_tail(&subdrv->list, &exynos_drm_subdrv_list);
  52. return 0;
  53. }
  54. EXPORT_SYMBOL_GPL(exynos_drm_subdrv_register);
  55. int exynos_drm_subdrv_unregister(struct exynos_drm_subdrv *subdrv)
  56. {
  57. if (!subdrv)
  58. return -EINVAL;
  59. list_del(&subdrv->list);
  60. return 0;
  61. }
  62. EXPORT_SYMBOL_GPL(exynos_drm_subdrv_unregister);
  63. int exynos_drm_device_subdrv_probe(struct drm_device *dev)
  64. {
  65. struct exynos_drm_subdrv *subdrv, *n;
  66. int err;
  67. if (!dev)
  68. return -EINVAL;
  69. list_for_each_entry_safe(subdrv, n, &exynos_drm_subdrv_list, list) {
  70. if (subdrv->probe) {
  71. subdrv->drm_dev = dev;
  72. /*
  73. * this probe callback would be called by sub driver
  74. * after setting of all resources to this sub driver,
  75. * such as clock, irq and register map are done.
  76. */
  77. err = subdrv->probe(dev, subdrv->dev);
  78. if (err) {
  79. DRM_DEBUG("exynos drm subdrv probe failed.\n");
  80. list_del(&subdrv->list);
  81. continue;
  82. }
  83. }
  84. }
  85. return 0;
  86. }
  87. EXPORT_SYMBOL_GPL(exynos_drm_device_subdrv_probe);
  88. int exynos_drm_device_subdrv_remove(struct drm_device *dev)
  89. {
  90. struct exynos_drm_subdrv *subdrv;
  91. if (!dev) {
  92. WARN(1, "Unexpected drm device unregister!\n");
  93. return -EINVAL;
  94. }
  95. list_for_each_entry(subdrv, &exynos_drm_subdrv_list, list) {
  96. if (subdrv->remove)
  97. subdrv->remove(dev, subdrv->dev);
  98. }
  99. return 0;
  100. }
  101. EXPORT_SYMBOL_GPL(exynos_drm_device_subdrv_remove);
  102. int exynos_drm_subdrv_open(struct drm_device *dev, struct drm_file *file)
  103. {
  104. struct exynos_drm_subdrv *subdrv;
  105. int ret;
  106. list_for_each_entry(subdrv, &exynos_drm_subdrv_list, list) {
  107. if (subdrv->open) {
  108. ret = subdrv->open(dev, subdrv->dev, file);
  109. if (ret)
  110. goto err;
  111. }
  112. }
  113. return 0;
  114. err:
  115. list_for_each_entry_reverse(subdrv, &subdrv->list, list) {
  116. if (subdrv->close)
  117. subdrv->close(dev, subdrv->dev, file);
  118. }
  119. return ret;
  120. }
  121. EXPORT_SYMBOL_GPL(exynos_drm_subdrv_open);
  122. void exynos_drm_subdrv_close(struct drm_device *dev, struct drm_file *file)
  123. {
  124. struct exynos_drm_subdrv *subdrv;
  125. list_for_each_entry(subdrv, &exynos_drm_subdrv_list, list) {
  126. if (subdrv->close)
  127. subdrv->close(dev, subdrv->dev, file);
  128. }
  129. }
  130. EXPORT_SYMBOL_GPL(exynos_drm_subdrv_close);