drm_mode_config.c 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478
  1. /*
  2. * Copyright (c) 2016 Intel Corporation
  3. *
  4. * Permission to use, copy, modify, distribute, and sell this software and its
  5. * documentation for any purpose is hereby granted without fee, provided that
  6. * the above copyright notice appear in all copies and that both that copyright
  7. * notice and this permission notice appear in supporting documentation, and
  8. * that the name of the copyright holders not be used in advertising or
  9. * publicity pertaining to distribution of the software without specific,
  10. * written prior permission. The copyright holders make no representations
  11. * about the suitability of this software for any purpose. It is provided "as
  12. * is" without express or implied warranty.
  13. *
  14. * THE COPYRIGHT HOLDERS DISCLAIM ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
  15. * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO
  16. * EVENT SHALL THE COPYRIGHT HOLDERS BE LIABLE FOR ANY SPECIAL, INDIRECT OR
  17. * CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE,
  18. * DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
  19. * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE
  20. * OF THIS SOFTWARE.
  21. */
  22. #include <drm/drm_encoder.h>
  23. #include <drm/drm_mode_config.h>
  24. #include <drm/drmP.h>
  25. #include "drm_crtc_internal.h"
  26. #include "drm_internal.h"
  27. int drm_modeset_register_all(struct drm_device *dev)
  28. {
  29. int ret;
  30. ret = drm_plane_register_all(dev);
  31. if (ret)
  32. goto err_plane;
  33. ret = drm_crtc_register_all(dev);
  34. if (ret)
  35. goto err_crtc;
  36. ret = drm_encoder_register_all(dev);
  37. if (ret)
  38. goto err_encoder;
  39. ret = drm_connector_register_all(dev);
  40. if (ret)
  41. goto err_connector;
  42. return 0;
  43. err_connector:
  44. drm_encoder_unregister_all(dev);
  45. err_encoder:
  46. drm_crtc_unregister_all(dev);
  47. err_crtc:
  48. drm_plane_unregister_all(dev);
  49. err_plane:
  50. return ret;
  51. }
  52. void drm_modeset_unregister_all(struct drm_device *dev)
  53. {
  54. drm_connector_unregister_all(dev);
  55. drm_encoder_unregister_all(dev);
  56. drm_crtc_unregister_all(dev);
  57. drm_plane_unregister_all(dev);
  58. }
  59. /**
  60. * drm_mode_getresources - get graphics configuration
  61. * @dev: drm device for the ioctl
  62. * @data: data pointer for the ioctl
  63. * @file_priv: drm file for the ioctl call
  64. *
  65. * Construct a set of configuration description structures and return
  66. * them to the user, including CRTC, connector and framebuffer configuration.
  67. *
  68. * Called by the user via ioctl.
  69. *
  70. * Returns:
  71. * Zero on success, negative errno on failure.
  72. */
  73. int drm_mode_getresources(struct drm_device *dev, void *data,
  74. struct drm_file *file_priv)
  75. {
  76. struct drm_mode_card_res *card_res = data;
  77. struct drm_framebuffer *fb;
  78. struct drm_connector *connector;
  79. struct drm_crtc *crtc;
  80. struct drm_encoder *encoder;
  81. int count, ret = 0;
  82. uint32_t __user *fb_id;
  83. uint32_t __user *crtc_id;
  84. uint32_t __user *connector_id;
  85. uint32_t __user *encoder_id;
  86. struct drm_connector_list_iter conn_iter;
  87. if (!drm_core_check_feature(dev, DRIVER_MODESET))
  88. return -EINVAL;
  89. mutex_lock(&file_priv->fbs_lock);
  90. count = 0;
  91. fb_id = u64_to_user_ptr(card_res->fb_id_ptr);
  92. list_for_each_entry(fb, &file_priv->fbs, filp_head) {
  93. if (count < card_res->count_fbs &&
  94. put_user(fb->base.id, fb_id + count)) {
  95. mutex_unlock(&file_priv->fbs_lock);
  96. return -EFAULT;
  97. }
  98. count++;
  99. }
  100. card_res->count_fbs = count;
  101. mutex_unlock(&file_priv->fbs_lock);
  102. card_res->max_height = dev->mode_config.max_height;
  103. card_res->min_height = dev->mode_config.min_height;
  104. card_res->max_width = dev->mode_config.max_width;
  105. card_res->min_width = dev->mode_config.min_width;
  106. count = 0;
  107. crtc_id = u64_to_user_ptr(card_res->crtc_id_ptr);
  108. drm_for_each_crtc(crtc, dev) {
  109. if (drm_lease_held(file_priv, crtc->base.id)) {
  110. if (count < card_res->count_crtcs &&
  111. put_user(crtc->base.id, crtc_id + count))
  112. return -EFAULT;
  113. count++;
  114. }
  115. }
  116. card_res->count_crtcs = count;
  117. count = 0;
  118. encoder_id = u64_to_user_ptr(card_res->encoder_id_ptr);
  119. drm_for_each_encoder(encoder, dev) {
  120. if (count < card_res->count_encoders &&
  121. put_user(encoder->base.id, encoder_id + count))
  122. return -EFAULT;
  123. count++;
  124. }
  125. card_res->count_encoders = count;
  126. drm_connector_list_iter_begin(dev, &conn_iter);
  127. count = 0;
  128. connector_id = u64_to_user_ptr(card_res->connector_id_ptr);
  129. drm_for_each_connector_iter(connector, &conn_iter) {
  130. if (drm_lease_held(file_priv, connector->base.id)) {
  131. if (count < card_res->count_connectors &&
  132. put_user(connector->base.id, connector_id + count)) {
  133. drm_connector_list_iter_end(&conn_iter);
  134. return -EFAULT;
  135. }
  136. count++;
  137. }
  138. }
  139. card_res->count_connectors = count;
  140. drm_connector_list_iter_end(&conn_iter);
  141. return ret;
  142. }
  143. /**
  144. * drm_mode_config_reset - call ->reset callbacks
  145. * @dev: drm device
  146. *
  147. * This functions calls all the crtc's, encoder's and connector's ->reset
  148. * callback. Drivers can use this in e.g. their driver load or resume code to
  149. * reset hardware and software state.
  150. */
  151. void drm_mode_config_reset(struct drm_device *dev)
  152. {
  153. struct drm_crtc *crtc;
  154. struct drm_plane *plane;
  155. struct drm_encoder *encoder;
  156. struct drm_connector *connector;
  157. struct drm_connector_list_iter conn_iter;
  158. drm_for_each_plane(plane, dev)
  159. if (plane->funcs->reset)
  160. plane->funcs->reset(plane);
  161. drm_for_each_crtc(crtc, dev)
  162. if (crtc->funcs->reset)
  163. crtc->funcs->reset(crtc);
  164. drm_for_each_encoder(encoder, dev)
  165. if (encoder->funcs->reset)
  166. encoder->funcs->reset(encoder);
  167. drm_connector_list_iter_begin(dev, &conn_iter);
  168. drm_for_each_connector_iter(connector, &conn_iter)
  169. if (connector->funcs->reset)
  170. connector->funcs->reset(connector);
  171. drm_connector_list_iter_end(&conn_iter);
  172. }
  173. EXPORT_SYMBOL(drm_mode_config_reset);
  174. /*
  175. * Global properties
  176. */
  177. static const struct drm_prop_enum_list drm_plane_type_enum_list[] = {
  178. { DRM_PLANE_TYPE_OVERLAY, "Overlay" },
  179. { DRM_PLANE_TYPE_PRIMARY, "Primary" },
  180. { DRM_PLANE_TYPE_CURSOR, "Cursor" },
  181. };
  182. static int drm_mode_create_standard_properties(struct drm_device *dev)
  183. {
  184. struct drm_property *prop;
  185. int ret;
  186. ret = drm_connector_create_standard_properties(dev);
  187. if (ret)
  188. return ret;
  189. prop = drm_property_create_enum(dev, DRM_MODE_PROP_IMMUTABLE,
  190. "type", drm_plane_type_enum_list,
  191. ARRAY_SIZE(drm_plane_type_enum_list));
  192. if (!prop)
  193. return -ENOMEM;
  194. dev->mode_config.plane_type_property = prop;
  195. prop = drm_property_create_range(dev, DRM_MODE_PROP_ATOMIC,
  196. "SRC_X", 0, UINT_MAX);
  197. if (!prop)
  198. return -ENOMEM;
  199. dev->mode_config.prop_src_x = prop;
  200. prop = drm_property_create_range(dev, DRM_MODE_PROP_ATOMIC,
  201. "SRC_Y", 0, UINT_MAX);
  202. if (!prop)
  203. return -ENOMEM;
  204. dev->mode_config.prop_src_y = prop;
  205. prop = drm_property_create_range(dev, DRM_MODE_PROP_ATOMIC,
  206. "SRC_W", 0, UINT_MAX);
  207. if (!prop)
  208. return -ENOMEM;
  209. dev->mode_config.prop_src_w = prop;
  210. prop = drm_property_create_range(dev, DRM_MODE_PROP_ATOMIC,
  211. "SRC_H", 0, UINT_MAX);
  212. if (!prop)
  213. return -ENOMEM;
  214. dev->mode_config.prop_src_h = prop;
  215. prop = drm_property_create_signed_range(dev, DRM_MODE_PROP_ATOMIC,
  216. "CRTC_X", INT_MIN, INT_MAX);
  217. if (!prop)
  218. return -ENOMEM;
  219. dev->mode_config.prop_crtc_x = prop;
  220. prop = drm_property_create_signed_range(dev, DRM_MODE_PROP_ATOMIC,
  221. "CRTC_Y", INT_MIN, INT_MAX);
  222. if (!prop)
  223. return -ENOMEM;
  224. dev->mode_config.prop_crtc_y = prop;
  225. prop = drm_property_create_range(dev, DRM_MODE_PROP_ATOMIC,
  226. "CRTC_W", 0, INT_MAX);
  227. if (!prop)
  228. return -ENOMEM;
  229. dev->mode_config.prop_crtc_w = prop;
  230. prop = drm_property_create_range(dev, DRM_MODE_PROP_ATOMIC,
  231. "CRTC_H", 0, INT_MAX);
  232. if (!prop)
  233. return -ENOMEM;
  234. dev->mode_config.prop_crtc_h = prop;
  235. prop = drm_property_create_object(dev, DRM_MODE_PROP_ATOMIC,
  236. "FB_ID", DRM_MODE_OBJECT_FB);
  237. if (!prop)
  238. return -ENOMEM;
  239. dev->mode_config.prop_fb_id = prop;
  240. prop = drm_property_create_signed_range(dev, DRM_MODE_PROP_ATOMIC,
  241. "IN_FENCE_FD", -1, INT_MAX);
  242. if (!prop)
  243. return -ENOMEM;
  244. dev->mode_config.prop_in_fence_fd = prop;
  245. prop = drm_property_create_range(dev, DRM_MODE_PROP_ATOMIC,
  246. "OUT_FENCE_PTR", 0, U64_MAX);
  247. if (!prop)
  248. return -ENOMEM;
  249. dev->mode_config.prop_out_fence_ptr = prop;
  250. prop = drm_property_create_object(dev, DRM_MODE_PROP_ATOMIC,
  251. "CRTC_ID", DRM_MODE_OBJECT_CRTC);
  252. if (!prop)
  253. return -ENOMEM;
  254. dev->mode_config.prop_crtc_id = prop;
  255. prop = drm_property_create_bool(dev, DRM_MODE_PROP_ATOMIC,
  256. "ACTIVE");
  257. if (!prop)
  258. return -ENOMEM;
  259. dev->mode_config.prop_active = prop;
  260. prop = drm_property_create(dev,
  261. DRM_MODE_PROP_ATOMIC | DRM_MODE_PROP_BLOB,
  262. "MODE_ID", 0);
  263. if (!prop)
  264. return -ENOMEM;
  265. dev->mode_config.prop_mode_id = prop;
  266. prop = drm_property_create(dev,
  267. DRM_MODE_PROP_BLOB,
  268. "DEGAMMA_LUT", 0);
  269. if (!prop)
  270. return -ENOMEM;
  271. dev->mode_config.degamma_lut_property = prop;
  272. prop = drm_property_create_range(dev,
  273. DRM_MODE_PROP_IMMUTABLE,
  274. "DEGAMMA_LUT_SIZE", 0, UINT_MAX);
  275. if (!prop)
  276. return -ENOMEM;
  277. dev->mode_config.degamma_lut_size_property = prop;
  278. prop = drm_property_create(dev,
  279. DRM_MODE_PROP_BLOB,
  280. "CTM", 0);
  281. if (!prop)
  282. return -ENOMEM;
  283. dev->mode_config.ctm_property = prop;
  284. prop = drm_property_create(dev,
  285. DRM_MODE_PROP_BLOB,
  286. "GAMMA_LUT", 0);
  287. if (!prop)
  288. return -ENOMEM;
  289. dev->mode_config.gamma_lut_property = prop;
  290. prop = drm_property_create_range(dev,
  291. DRM_MODE_PROP_IMMUTABLE,
  292. "GAMMA_LUT_SIZE", 0, UINT_MAX);
  293. if (!prop)
  294. return -ENOMEM;
  295. dev->mode_config.gamma_lut_size_property = prop;
  296. prop = drm_property_create(dev,
  297. DRM_MODE_PROP_IMMUTABLE | DRM_MODE_PROP_BLOB,
  298. "IN_FORMATS", 0);
  299. if (!prop)
  300. return -ENOMEM;
  301. dev->mode_config.modifiers_property = prop;
  302. return 0;
  303. }
  304. /**
  305. * drm_mode_config_init - initialize DRM mode_configuration structure
  306. * @dev: DRM device
  307. *
  308. * Initialize @dev's mode_config structure, used for tracking the graphics
  309. * configuration of @dev.
  310. *
  311. * Since this initializes the modeset locks, no locking is possible. Which is no
  312. * problem, since this should happen single threaded at init time. It is the
  313. * driver's problem to ensure this guarantee.
  314. *
  315. */
  316. void drm_mode_config_init(struct drm_device *dev)
  317. {
  318. mutex_init(&dev->mode_config.mutex);
  319. drm_modeset_lock_init(&dev->mode_config.connection_mutex);
  320. mutex_init(&dev->mode_config.idr_mutex);
  321. mutex_init(&dev->mode_config.fb_lock);
  322. mutex_init(&dev->mode_config.blob_lock);
  323. INIT_LIST_HEAD(&dev->mode_config.fb_list);
  324. INIT_LIST_HEAD(&dev->mode_config.crtc_list);
  325. INIT_LIST_HEAD(&dev->mode_config.connector_list);
  326. INIT_LIST_HEAD(&dev->mode_config.encoder_list);
  327. INIT_LIST_HEAD(&dev->mode_config.property_list);
  328. INIT_LIST_HEAD(&dev->mode_config.property_blob_list);
  329. INIT_LIST_HEAD(&dev->mode_config.plane_list);
  330. idr_init(&dev->mode_config.crtc_idr);
  331. idr_init(&dev->mode_config.tile_idr);
  332. ida_init(&dev->mode_config.connector_ida);
  333. spin_lock_init(&dev->mode_config.connector_list_lock);
  334. drm_mode_create_standard_properties(dev);
  335. /* Just to be sure */
  336. dev->mode_config.num_fb = 0;
  337. dev->mode_config.num_connector = 0;
  338. dev->mode_config.num_crtc = 0;
  339. dev->mode_config.num_encoder = 0;
  340. dev->mode_config.num_total_plane = 0;
  341. }
  342. EXPORT_SYMBOL(drm_mode_config_init);
  343. /**
  344. * drm_mode_config_cleanup - free up DRM mode_config info
  345. * @dev: DRM device
  346. *
  347. * Free up all the connectors and CRTCs associated with this DRM device, then
  348. * free up the framebuffers and associated buffer objects.
  349. *
  350. * Note that since this /should/ happen single-threaded at driver/device
  351. * teardown time, no locking is required. It's the driver's job to ensure that
  352. * this guarantee actually holds true.
  353. *
  354. * FIXME: cleanup any dangling user buffer objects too
  355. */
  356. void drm_mode_config_cleanup(struct drm_device *dev)
  357. {
  358. struct drm_connector *connector;
  359. struct drm_connector_list_iter conn_iter;
  360. struct drm_crtc *crtc, *ct;
  361. struct drm_encoder *encoder, *enct;
  362. struct drm_framebuffer *fb, *fbt;
  363. struct drm_property *property, *pt;
  364. struct drm_property_blob *blob, *bt;
  365. struct drm_plane *plane, *plt;
  366. list_for_each_entry_safe(encoder, enct, &dev->mode_config.encoder_list,
  367. head) {
  368. encoder->funcs->destroy(encoder);
  369. }
  370. drm_connector_list_iter_begin(dev, &conn_iter);
  371. drm_for_each_connector_iter(connector, &conn_iter) {
  372. /* drm_connector_list_iter holds an full reference to the
  373. * current connector itself, which means it is inherently safe
  374. * against unreferencing the current connector - but not against
  375. * deleting it right away. */
  376. drm_connector_put(connector);
  377. }
  378. drm_connector_list_iter_end(&conn_iter);
  379. if (WARN_ON(!list_empty(&dev->mode_config.connector_list))) {
  380. drm_connector_list_iter_begin(dev, &conn_iter);
  381. drm_for_each_connector_iter(connector, &conn_iter)
  382. DRM_ERROR("connector %s leaked!\n", connector->name);
  383. drm_connector_list_iter_end(&conn_iter);
  384. }
  385. list_for_each_entry_safe(property, pt, &dev->mode_config.property_list,
  386. head) {
  387. drm_property_destroy(dev, property);
  388. }
  389. list_for_each_entry_safe(plane, plt, &dev->mode_config.plane_list,
  390. head) {
  391. plane->funcs->destroy(plane);
  392. }
  393. list_for_each_entry_safe(crtc, ct, &dev->mode_config.crtc_list, head) {
  394. crtc->funcs->destroy(crtc);
  395. }
  396. list_for_each_entry_safe(blob, bt, &dev->mode_config.property_blob_list,
  397. head_global) {
  398. drm_property_blob_put(blob);
  399. }
  400. /*
  401. * Single-threaded teardown context, so it's not required to grab the
  402. * fb_lock to protect against concurrent fb_list access. Contrary, it
  403. * would actually deadlock with the drm_framebuffer_cleanup function.
  404. *
  405. * Also, if there are any framebuffers left, that's a driver leak now,
  406. * so politely WARN about this.
  407. */
  408. WARN_ON(!list_empty(&dev->mode_config.fb_list));
  409. list_for_each_entry_safe(fb, fbt, &dev->mode_config.fb_list, head) {
  410. drm_framebuffer_free(&fb->base.refcount);
  411. }
  412. ida_destroy(&dev->mode_config.connector_ida);
  413. idr_destroy(&dev->mode_config.tile_idr);
  414. idr_destroy(&dev->mode_config.crtc_idr);
  415. drm_modeset_lock_fini(&dev->mode_config.connection_mutex);
  416. }
  417. EXPORT_SYMBOL(drm_mode_config_cleanup);