drm_fourcc.c 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458
  1. /*
  2. * Copyright (c) 2016 Laurent Pinchart <[email protected]>
  3. *
  4. * DRM core format related functions
  5. *
  6. * Permission to use, copy, modify, distribute, and sell this software and its
  7. * documentation for any purpose is hereby granted without fee, provided that
  8. * the above copyright notice appear in all copies and that both that copyright
  9. * notice and this permission notice appear in supporting documentation, and
  10. * that the name of the copyright holders not be used in advertising or
  11. * publicity pertaining to distribution of the software without specific,
  12. * written prior permission. The copyright holders make no representations
  13. * about the suitability of this software for any purpose. It is provided "as
  14. * is" without express or implied warranty.
  15. *
  16. * THE COPYRIGHT HOLDERS DISCLAIM ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
  17. * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO
  18. * EVENT SHALL THE COPYRIGHT HOLDERS BE LIABLE FOR ANY SPECIAL, INDIRECT OR
  19. * CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE,
  20. * DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
  21. * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE
  22. * OF THIS SOFTWARE.
  23. */
  24. #include <linux/bug.h>
  25. #include <linux/ctype.h>
  26. #include <linux/export.h>
  27. #include <linux/kernel.h>
  28. #include <drm/drm_device.h>
  29. #include <drm/drm_fourcc.h>
  30. /**
  31. * drm_mode_legacy_fb_format - compute drm fourcc code from legacy description
  32. * @bpp: bits per pixels
  33. * @depth: bit depth per pixel
  34. *
  35. * Computes a drm fourcc pixel format code for the given @bpp/@depth values.
  36. * Useful in fbdev emulation code, since that deals in those values.
  37. */
  38. uint32_t drm_mode_legacy_fb_format(uint32_t bpp, uint32_t depth)
  39. {
  40. uint32_t fmt = DRM_FORMAT_INVALID;
  41. switch (bpp) {
  42. case 1:
  43. if (depth == 1)
  44. fmt = DRM_FORMAT_C1;
  45. break;
  46. case 2:
  47. if (depth == 2)
  48. fmt = DRM_FORMAT_C2;
  49. break;
  50. case 4:
  51. if (depth == 4)
  52. fmt = DRM_FORMAT_C4;
  53. break;
  54. case 8:
  55. if (depth == 8)
  56. fmt = DRM_FORMAT_C8;
  57. break;
  58. case 16:
  59. switch (depth) {
  60. case 15:
  61. fmt = DRM_FORMAT_XRGB1555;
  62. break;
  63. case 16:
  64. fmt = DRM_FORMAT_RGB565;
  65. break;
  66. default:
  67. break;
  68. }
  69. break;
  70. case 24:
  71. if (depth == 24)
  72. fmt = DRM_FORMAT_RGB888;
  73. break;
  74. case 32:
  75. switch (depth) {
  76. case 24:
  77. fmt = DRM_FORMAT_XRGB8888;
  78. break;
  79. case 30:
  80. fmt = DRM_FORMAT_XRGB2101010;
  81. break;
  82. case 32:
  83. fmt = DRM_FORMAT_ARGB8888;
  84. break;
  85. default:
  86. break;
  87. }
  88. break;
  89. default:
  90. break;
  91. }
  92. return fmt;
  93. }
  94. EXPORT_SYMBOL(drm_mode_legacy_fb_format);
  95. /**
  96. * drm_driver_legacy_fb_format - compute drm fourcc code from legacy description
  97. * @dev: DRM device
  98. * @bpp: bits per pixels
  99. * @depth: bit depth per pixel
  100. *
  101. * Computes a drm fourcc pixel format code for the given @bpp/@depth values.
  102. * Unlike drm_mode_legacy_fb_format() this looks at the drivers mode_config,
  103. * and depending on the &drm_mode_config.quirk_addfb_prefer_host_byte_order flag
  104. * it returns little endian byte order or host byte order framebuffer formats.
  105. */
  106. uint32_t drm_driver_legacy_fb_format(struct drm_device *dev,
  107. uint32_t bpp, uint32_t depth)
  108. {
  109. uint32_t fmt = drm_mode_legacy_fb_format(bpp, depth);
  110. if (dev->mode_config.quirk_addfb_prefer_host_byte_order) {
  111. if (fmt == DRM_FORMAT_XRGB8888)
  112. fmt = DRM_FORMAT_HOST_XRGB8888;
  113. if (fmt == DRM_FORMAT_ARGB8888)
  114. fmt = DRM_FORMAT_HOST_ARGB8888;
  115. if (fmt == DRM_FORMAT_RGB565)
  116. fmt = DRM_FORMAT_HOST_RGB565;
  117. if (fmt == DRM_FORMAT_XRGB1555)
  118. fmt = DRM_FORMAT_HOST_XRGB1555;
  119. }
  120. if (dev->mode_config.quirk_addfb_prefer_xbgr_30bpp &&
  121. fmt == DRM_FORMAT_XRGB2101010)
  122. fmt = DRM_FORMAT_XBGR2101010;
  123. return fmt;
  124. }
  125. EXPORT_SYMBOL(drm_driver_legacy_fb_format);
  126. /*
  127. * Internal function to query information for a given format. See
  128. * drm_format_info() for the public API.
  129. */
  130. const struct drm_format_info *__drm_format_info(u32 format)
  131. {
  132. static const struct drm_format_info formats[] = {
  133. { .format = DRM_FORMAT_C1, .depth = 1, .num_planes = 1,
  134. .char_per_block = { 1, }, .block_w = { 8, }, .block_h = { 1, }, .hsub = 1, .vsub = 1, .is_color_indexed = true },
  135. { .format = DRM_FORMAT_C2, .depth = 2, .num_planes = 1,
  136. .char_per_block = { 1, }, .block_w = { 4, }, .block_h = { 1, }, .hsub = 1, .vsub = 1, .is_color_indexed = true },
  137. { .format = DRM_FORMAT_C4, .depth = 4, .num_planes = 1,
  138. .char_per_block = { 1, }, .block_w = { 2, }, .block_h = { 1, }, .hsub = 1, .vsub = 1, .is_color_indexed = true },
  139. { .format = DRM_FORMAT_C8, .depth = 8, .num_planes = 1, .cpp = { 1, 0, 0 }, .hsub = 1, .vsub = 1, .is_color_indexed = true },
  140. { .format = DRM_FORMAT_D1, .depth = 1, .num_planes = 1,
  141. .char_per_block = { 1, }, .block_w = { 8, }, .block_h = { 1, }, .hsub = 1, .vsub = 1 },
  142. { .format = DRM_FORMAT_D2, .depth = 2, .num_planes = 1,
  143. .char_per_block = { 1, }, .block_w = { 4, }, .block_h = { 1, }, .hsub = 1, .vsub = 1 },
  144. { .format = DRM_FORMAT_D4, .depth = 4, .num_planes = 1,
  145. .char_per_block = { 1, }, .block_w = { 2, }, .block_h = { 1, }, .hsub = 1, .vsub = 1 },
  146. { .format = DRM_FORMAT_D8, .depth = 8, .num_planes = 1, .cpp = { 1, 0, 0 }, .hsub = 1, .vsub = 1 },
  147. { .format = DRM_FORMAT_R1, .depth = 1, .num_planes = 1,
  148. .char_per_block = { 1, }, .block_w = { 8, }, .block_h = { 1, }, .hsub = 1, .vsub = 1 },
  149. { .format = DRM_FORMAT_R2, .depth = 2, .num_planes = 1,
  150. .char_per_block = { 1, }, .block_w = { 4, }, .block_h = { 1, }, .hsub = 1, .vsub = 1 },
  151. { .format = DRM_FORMAT_R4, .depth = 4, .num_planes = 1,
  152. .char_per_block = { 1, }, .block_w = { 2, }, .block_h = { 1, }, .hsub = 1, .vsub = 1 },
  153. { .format = DRM_FORMAT_R8, .depth = 8, .num_planes = 1, .cpp = { 1, 0, 0 }, .hsub = 1, .vsub = 1 },
  154. { .format = DRM_FORMAT_R10, .depth = 10, .num_planes = 1, .cpp = { 2, 0, 0 }, .hsub = 1, .vsub = 1 },
  155. { .format = DRM_FORMAT_R12, .depth = 12, .num_planes = 1, .cpp = { 2, 0, 0 }, .hsub = 1, .vsub = 1 },
  156. { .format = DRM_FORMAT_RGB332, .depth = 8, .num_planes = 1, .cpp = { 1, 0, 0 }, .hsub = 1, .vsub = 1 },
  157. { .format = DRM_FORMAT_BGR233, .depth = 8, .num_planes = 1, .cpp = { 1, 0, 0 }, .hsub = 1, .vsub = 1 },
  158. { .format = DRM_FORMAT_XRGB4444, .depth = 0, .num_planes = 1, .cpp = { 2, 0, 0 }, .hsub = 1, .vsub = 1 },
  159. { .format = DRM_FORMAT_XBGR4444, .depth = 0, .num_planes = 1, .cpp = { 2, 0, 0 }, .hsub = 1, .vsub = 1 },
  160. { .format = DRM_FORMAT_RGBX4444, .depth = 0, .num_planes = 1, .cpp = { 2, 0, 0 }, .hsub = 1, .vsub = 1 },
  161. { .format = DRM_FORMAT_BGRX4444, .depth = 0, .num_planes = 1, .cpp = { 2, 0, 0 }, .hsub = 1, .vsub = 1 },
  162. { .format = DRM_FORMAT_ARGB4444, .depth = 0, .num_planes = 1, .cpp = { 2, 0, 0 }, .hsub = 1, .vsub = 1, .has_alpha = true },
  163. { .format = DRM_FORMAT_ABGR4444, .depth = 0, .num_planes = 1, .cpp = { 2, 0, 0 }, .hsub = 1, .vsub = 1, .has_alpha = true },
  164. { .format = DRM_FORMAT_RGBA4444, .depth = 0, .num_planes = 1, .cpp = { 2, 0, 0 }, .hsub = 1, .vsub = 1, .has_alpha = true },
  165. { .format = DRM_FORMAT_BGRA4444, .depth = 0, .num_planes = 1, .cpp = { 2, 0, 0 }, .hsub = 1, .vsub = 1, .has_alpha = true },
  166. { .format = DRM_FORMAT_XRGB1555, .depth = 15, .num_planes = 1, .cpp = { 2, 0, 0 }, .hsub = 1, .vsub = 1 },
  167. { .format = DRM_FORMAT_XBGR1555, .depth = 15, .num_planes = 1, .cpp = { 2, 0, 0 }, .hsub = 1, .vsub = 1 },
  168. { .format = DRM_FORMAT_RGBX5551, .depth = 15, .num_planes = 1, .cpp = { 2, 0, 0 }, .hsub = 1, .vsub = 1 },
  169. { .format = DRM_FORMAT_BGRX5551, .depth = 15, .num_planes = 1, .cpp = { 2, 0, 0 }, .hsub = 1, .vsub = 1 },
  170. { .format = DRM_FORMAT_ARGB1555, .depth = 15, .num_planes = 1, .cpp = { 2, 0, 0 }, .hsub = 1, .vsub = 1, .has_alpha = true },
  171. { .format = DRM_FORMAT_ABGR1555, .depth = 15, .num_planes = 1, .cpp = { 2, 0, 0 }, .hsub = 1, .vsub = 1, .has_alpha = true },
  172. { .format = DRM_FORMAT_RGBA5551, .depth = 15, .num_planes = 1, .cpp = { 2, 0, 0 }, .hsub = 1, .vsub = 1, .has_alpha = true },
  173. { .format = DRM_FORMAT_BGRA5551, .depth = 15, .num_planes = 1, .cpp = { 2, 0, 0 }, .hsub = 1, .vsub = 1, .has_alpha = true },
  174. { .format = DRM_FORMAT_RGB565, .depth = 16, .num_planes = 1, .cpp = { 2, 0, 0 }, .hsub = 1, .vsub = 1 },
  175. { .format = DRM_FORMAT_BGR565, .depth = 16, .num_planes = 1, .cpp = { 2, 0, 0 }, .hsub = 1, .vsub = 1 },
  176. #ifdef __BIG_ENDIAN
  177. { .format = DRM_FORMAT_XRGB1555 | DRM_FORMAT_BIG_ENDIAN, .depth = 15, .num_planes = 1, .cpp = { 2, 0, 0 }, .hsub = 1, .vsub = 1 },
  178. { .format = DRM_FORMAT_RGB565 | DRM_FORMAT_BIG_ENDIAN, .depth = 16, .num_planes = 1, .cpp = { 2, 0, 0 }, .hsub = 1, .vsub = 1 },
  179. #endif
  180. { .format = DRM_FORMAT_RGB888, .depth = 24, .num_planes = 1, .cpp = { 3, 0, 0 }, .hsub = 1, .vsub = 1 },
  181. { .format = DRM_FORMAT_BGR888, .depth = 24, .num_planes = 1, .cpp = { 3, 0, 0 }, .hsub = 1, .vsub = 1 },
  182. { .format = DRM_FORMAT_XRGB8888, .depth = 24, .num_planes = 1, .cpp = { 4, 0, 0 }, .hsub = 1, .vsub = 1 },
  183. { .format = DRM_FORMAT_XBGR8888, .depth = 24, .num_planes = 1, .cpp = { 4, 0, 0 }, .hsub = 1, .vsub = 1 },
  184. { .format = DRM_FORMAT_RGBX8888, .depth = 24, .num_planes = 1, .cpp = { 4, 0, 0 }, .hsub = 1, .vsub = 1 },
  185. { .format = DRM_FORMAT_BGRX8888, .depth = 24, .num_planes = 1, .cpp = { 4, 0, 0 }, .hsub = 1, .vsub = 1 },
  186. { .format = DRM_FORMAT_RGB565_A8, .depth = 24, .num_planes = 2, .cpp = { 2, 1, 0 }, .hsub = 1, .vsub = 1, .has_alpha = true },
  187. { .format = DRM_FORMAT_BGR565_A8, .depth = 24, .num_planes = 2, .cpp = { 2, 1, 0 }, .hsub = 1, .vsub = 1, .has_alpha = true },
  188. { .format = DRM_FORMAT_XRGB2101010, .depth = 30, .num_planes = 1, .cpp = { 4, 0, 0 }, .hsub = 1, .vsub = 1 },
  189. { .format = DRM_FORMAT_XBGR2101010, .depth = 30, .num_planes = 1, .cpp = { 4, 0, 0 }, .hsub = 1, .vsub = 1 },
  190. { .format = DRM_FORMAT_RGBX1010102, .depth = 30, .num_planes = 1, .cpp = { 4, 0, 0 }, .hsub = 1, .vsub = 1 },
  191. { .format = DRM_FORMAT_BGRX1010102, .depth = 30, .num_planes = 1, .cpp = { 4, 0, 0 }, .hsub = 1, .vsub = 1 },
  192. { .format = DRM_FORMAT_ARGB2101010, .depth = 30, .num_planes = 1, .cpp = { 4, 0, 0 }, .hsub = 1, .vsub = 1, .has_alpha = true },
  193. { .format = DRM_FORMAT_ABGR2101010, .depth = 30, .num_planes = 1, .cpp = { 4, 0, 0 }, .hsub = 1, .vsub = 1, .has_alpha = true },
  194. { .format = DRM_FORMAT_RGBA1010102, .depth = 30, .num_planes = 1, .cpp = { 4, 0, 0 }, .hsub = 1, .vsub = 1, .has_alpha = true },
  195. { .format = DRM_FORMAT_BGRA1010102, .depth = 30, .num_planes = 1, .cpp = { 4, 0, 0 }, .hsub = 1, .vsub = 1, .has_alpha = true },
  196. { .format = DRM_FORMAT_ARGB8888, .depth = 32, .num_planes = 1, .cpp = { 4, 0, 0 }, .hsub = 1, .vsub = 1, .has_alpha = true },
  197. { .format = DRM_FORMAT_ABGR8888, .depth = 32, .num_planes = 1, .cpp = { 4, 0, 0 }, .hsub = 1, .vsub = 1, .has_alpha = true },
  198. { .format = DRM_FORMAT_RGBA8888, .depth = 32, .num_planes = 1, .cpp = { 4, 0, 0 }, .hsub = 1, .vsub = 1, .has_alpha = true },
  199. { .format = DRM_FORMAT_BGRA8888, .depth = 32, .num_planes = 1, .cpp = { 4, 0, 0 }, .hsub = 1, .vsub = 1, .has_alpha = true },
  200. { .format = DRM_FORMAT_XRGB16161616F, .depth = 0, .num_planes = 1, .cpp = { 8, 0, 0 }, .hsub = 1, .vsub = 1 },
  201. { .format = DRM_FORMAT_XBGR16161616F, .depth = 0, .num_planes = 1, .cpp = { 8, 0, 0 }, .hsub = 1, .vsub = 1 },
  202. { .format = DRM_FORMAT_ARGB16161616F, .depth = 0, .num_planes = 1, .cpp = { 8, 0, 0 }, .hsub = 1, .vsub = 1, .has_alpha = true },
  203. { .format = DRM_FORMAT_ABGR16161616F, .depth = 0, .num_planes = 1, .cpp = { 8, 0, 0 }, .hsub = 1, .vsub = 1, .has_alpha = true },
  204. { .format = DRM_FORMAT_AXBXGXRX106106106106, .depth = 0, .num_planes = 1, .cpp = { 8, 0, 0 }, .hsub = 1, .vsub = 1, .has_alpha = true },
  205. { .format = DRM_FORMAT_XRGB16161616, .depth = 0, .num_planes = 1, .cpp = { 8, 0, 0 }, .hsub = 1, .vsub = 1 },
  206. { .format = DRM_FORMAT_XBGR16161616, .depth = 0, .num_planes = 1, .cpp = { 8, 0, 0 }, .hsub = 1, .vsub = 1 },
  207. { .format = DRM_FORMAT_ARGB16161616, .depth = 0, .num_planes = 1, .cpp = { 8, 0, 0 }, .hsub = 1, .vsub = 1, .has_alpha = true },
  208. { .format = DRM_FORMAT_ABGR16161616, .depth = 0, .num_planes = 1, .cpp = { 8, 0, 0 }, .hsub = 1, .vsub = 1, .has_alpha = true },
  209. { .format = DRM_FORMAT_RGB888_A8, .depth = 32, .num_planes = 2, .cpp = { 3, 1, 0 }, .hsub = 1, .vsub = 1, .has_alpha = true },
  210. { .format = DRM_FORMAT_BGR888_A8, .depth = 32, .num_planes = 2, .cpp = { 3, 1, 0 }, .hsub = 1, .vsub = 1, .has_alpha = true },
  211. { .format = DRM_FORMAT_XRGB8888_A8, .depth = 32, .num_planes = 2, .cpp = { 4, 1, 0 }, .hsub = 1, .vsub = 1, .has_alpha = true },
  212. { .format = DRM_FORMAT_XBGR8888_A8, .depth = 32, .num_planes = 2, .cpp = { 4, 1, 0 }, .hsub = 1, .vsub = 1, .has_alpha = true },
  213. { .format = DRM_FORMAT_RGBX8888_A8, .depth = 32, .num_planes = 2, .cpp = { 4, 1, 0 }, .hsub = 1, .vsub = 1, .has_alpha = true },
  214. { .format = DRM_FORMAT_BGRX8888_A8, .depth = 32, .num_planes = 2, .cpp = { 4, 1, 0 }, .hsub = 1, .vsub = 1, .has_alpha = true },
  215. { .format = DRM_FORMAT_YUV410, .depth = 0, .num_planes = 3, .cpp = { 1, 1, 1 }, .hsub = 4, .vsub = 4, .is_yuv = true },
  216. { .format = DRM_FORMAT_YVU410, .depth = 0, .num_planes = 3, .cpp = { 1, 1, 1 }, .hsub = 4, .vsub = 4, .is_yuv = true },
  217. { .format = DRM_FORMAT_YUV411, .depth = 0, .num_planes = 3, .cpp = { 1, 1, 1 }, .hsub = 4, .vsub = 1, .is_yuv = true },
  218. { .format = DRM_FORMAT_YVU411, .depth = 0, .num_planes = 3, .cpp = { 1, 1, 1 }, .hsub = 4, .vsub = 1, .is_yuv = true },
  219. { .format = DRM_FORMAT_YUV420, .depth = 0, .num_planes = 3, .cpp = { 1, 1, 1 }, .hsub = 2, .vsub = 2, .is_yuv = true },
  220. { .format = DRM_FORMAT_YVU420, .depth = 0, .num_planes = 3, .cpp = { 1, 1, 1 }, .hsub = 2, .vsub = 2, .is_yuv = true },
  221. { .format = DRM_FORMAT_YUV422, .depth = 0, .num_planes = 3, .cpp = { 1, 1, 1 }, .hsub = 2, .vsub = 1, .is_yuv = true },
  222. { .format = DRM_FORMAT_YVU422, .depth = 0, .num_planes = 3, .cpp = { 1, 1, 1 }, .hsub = 2, .vsub = 1, .is_yuv = true },
  223. { .format = DRM_FORMAT_YUV444, .depth = 0, .num_planes = 3, .cpp = { 1, 1, 1 }, .hsub = 1, .vsub = 1, .is_yuv = true },
  224. { .format = DRM_FORMAT_YVU444, .depth = 0, .num_planes = 3, .cpp = { 1, 1, 1 }, .hsub = 1, .vsub = 1, .is_yuv = true },
  225. { .format = DRM_FORMAT_NV12, .depth = 0, .num_planes = 2, .cpp = { 1, 2, 0 }, .hsub = 2, .vsub = 2, .is_yuv = true },
  226. { .format = DRM_FORMAT_NV21, .depth = 0, .num_planes = 2, .cpp = { 1, 2, 0 }, .hsub = 2, .vsub = 2, .is_yuv = true },
  227. { .format = DRM_FORMAT_NV16, .depth = 0, .num_planes = 2, .cpp = { 1, 2, 0 }, .hsub = 2, .vsub = 1, .is_yuv = true },
  228. { .format = DRM_FORMAT_NV61, .depth = 0, .num_planes = 2, .cpp = { 1, 2, 0 }, .hsub = 2, .vsub = 1, .is_yuv = true },
  229. { .format = DRM_FORMAT_NV24, .depth = 0, .num_planes = 2, .cpp = { 1, 2, 0 }, .hsub = 1, .vsub = 1, .is_yuv = true },
  230. { .format = DRM_FORMAT_NV42, .depth = 0, .num_planes = 2, .cpp = { 1, 2, 0 }, .hsub = 1, .vsub = 1, .is_yuv = true },
  231. { .format = DRM_FORMAT_YUYV, .depth = 0, .num_planes = 1, .cpp = { 2, 0, 0 }, .hsub = 2, .vsub = 1, .is_yuv = true },
  232. { .format = DRM_FORMAT_YVYU, .depth = 0, .num_planes = 1, .cpp = { 2, 0, 0 }, .hsub = 2, .vsub = 1, .is_yuv = true },
  233. { .format = DRM_FORMAT_UYVY, .depth = 0, .num_planes = 1, .cpp = { 2, 0, 0 }, .hsub = 2, .vsub = 1, .is_yuv = true },
  234. { .format = DRM_FORMAT_VYUY, .depth = 0, .num_planes = 1, .cpp = { 2, 0, 0 }, .hsub = 2, .vsub = 1, .is_yuv = true },
  235. { .format = DRM_FORMAT_XYUV8888, .depth = 0, .num_planes = 1, .cpp = { 4, 0, 0 }, .hsub = 1, .vsub = 1, .is_yuv = true },
  236. { .format = DRM_FORMAT_VUY888, .depth = 0, .num_planes = 1, .cpp = { 3, 0, 0 }, .hsub = 1, .vsub = 1, .is_yuv = true },
  237. { .format = DRM_FORMAT_AYUV, .depth = 0, .num_planes = 1, .cpp = { 4, 0, 0 }, .hsub = 1, .vsub = 1, .has_alpha = true, .is_yuv = true },
  238. { .format = DRM_FORMAT_Y210, .depth = 0, .num_planes = 1, .cpp = { 4, 0, 0 }, .hsub = 2, .vsub = 1, .is_yuv = true },
  239. { .format = DRM_FORMAT_Y212, .depth = 0, .num_planes = 1, .cpp = { 4, 0, 0 }, .hsub = 2, .vsub = 1, .is_yuv = true },
  240. { .format = DRM_FORMAT_Y216, .depth = 0, .num_planes = 1, .cpp = { 4, 0, 0 }, .hsub = 2, .vsub = 1, .is_yuv = true },
  241. { .format = DRM_FORMAT_Y410, .depth = 0, .num_planes = 1, .cpp = { 4, 0, 0 }, .hsub = 1, .vsub = 1, .has_alpha = true, .is_yuv = true },
  242. { .format = DRM_FORMAT_Y412, .depth = 0, .num_planes = 1, .cpp = { 8, 0, 0 }, .hsub = 1, .vsub = 1, .has_alpha = true, .is_yuv = true },
  243. { .format = DRM_FORMAT_Y416, .depth = 0, .num_planes = 1, .cpp = { 8, 0, 0 }, .hsub = 1, .vsub = 1, .has_alpha = true, .is_yuv = true },
  244. { .format = DRM_FORMAT_XVYU2101010, .depth = 0, .num_planes = 1, .cpp = { 4, 0, 0 }, .hsub = 1, .vsub = 1, .is_yuv = true },
  245. { .format = DRM_FORMAT_XVYU12_16161616, .depth = 0, .num_planes = 1, .cpp = { 8, 0, 0 }, .hsub = 1, .vsub = 1, .is_yuv = true },
  246. { .format = DRM_FORMAT_XVYU16161616, .depth = 0, .num_planes = 1, .cpp = { 8, 0, 0 }, .hsub = 1, .vsub = 1, .is_yuv = true },
  247. { .format = DRM_FORMAT_Y0L0, .depth = 0, .num_planes = 1,
  248. .char_per_block = { 8, 0, 0 }, .block_w = { 2, 0, 0 }, .block_h = { 2, 0, 0 },
  249. .hsub = 2, .vsub = 2, .has_alpha = true, .is_yuv = true },
  250. { .format = DRM_FORMAT_X0L0, .depth = 0, .num_planes = 1,
  251. .char_per_block = { 8, 0, 0 }, .block_w = { 2, 0, 0 }, .block_h = { 2, 0, 0 },
  252. .hsub = 2, .vsub = 2, .is_yuv = true },
  253. { .format = DRM_FORMAT_Y0L2, .depth = 0, .num_planes = 1,
  254. .char_per_block = { 8, 0, 0 }, .block_w = { 2, 0, 0 }, .block_h = { 2, 0, 0 },
  255. .hsub = 2, .vsub = 2, .has_alpha = true, .is_yuv = true },
  256. { .format = DRM_FORMAT_X0L2, .depth = 0, .num_planes = 1,
  257. .char_per_block = { 8, 0, 0 }, .block_w = { 2, 0, 0 }, .block_h = { 2, 0, 0 },
  258. .hsub = 2, .vsub = 2, .is_yuv = true },
  259. { .format = DRM_FORMAT_P010, .depth = 0, .num_planes = 2,
  260. .char_per_block = { 2, 4, 0 }, .block_w = { 1, 1, 0 }, .block_h = { 1, 1, 0 },
  261. .hsub = 2, .vsub = 2, .is_yuv = true},
  262. { .format = DRM_FORMAT_P012, .depth = 0, .num_planes = 2,
  263. .char_per_block = { 2, 4, 0 }, .block_w = { 1, 1, 0 }, .block_h = { 1, 1, 0 },
  264. .hsub = 2, .vsub = 2, .is_yuv = true},
  265. { .format = DRM_FORMAT_P016, .depth = 0, .num_planes = 2,
  266. .char_per_block = { 2, 4, 0 }, .block_w = { 1, 1, 0 }, .block_h = { 1, 1, 0 },
  267. .hsub = 2, .vsub = 2, .is_yuv = true},
  268. { .format = DRM_FORMAT_P210, .depth = 0,
  269. .num_planes = 2, .char_per_block = { 2, 4, 0 },
  270. .block_w = { 1, 1, 0 }, .block_h = { 1, 1, 0 }, .hsub = 2,
  271. .vsub = 1, .is_yuv = true },
  272. { .format = DRM_FORMAT_VUY101010, .depth = 0,
  273. .num_planes = 1, .cpp = { 0, 0, 0 }, .hsub = 1, .vsub = 1,
  274. .is_yuv = true },
  275. { .format = DRM_FORMAT_YUV420_8BIT, .depth = 0,
  276. .num_planes = 1, .cpp = { 0, 0, 0 }, .hsub = 2, .vsub = 2,
  277. .is_yuv = true },
  278. { .format = DRM_FORMAT_YUV420_10BIT, .depth = 0,
  279. .num_planes = 1, .cpp = { 0, 0, 0 }, .hsub = 2, .vsub = 2,
  280. .is_yuv = true },
  281. { .format = DRM_FORMAT_NV15, .depth = 0,
  282. .num_planes = 2, .char_per_block = { 5, 5, 0 },
  283. .block_w = { 4, 2, 0 }, .block_h = { 1, 1, 0 }, .hsub = 2,
  284. .vsub = 2, .is_yuv = true },
  285. { .format = DRM_FORMAT_NV20, .depth = 0,
  286. .num_planes = 2, .char_per_block = { 5, 5, 0 },
  287. .block_w = { 4, 2, 0 }, .block_h = { 1, 1, 0 }, .hsub = 2,
  288. .vsub = 1, .is_yuv = true },
  289. { .format = DRM_FORMAT_NV30, .depth = 0,
  290. .num_planes = 2, .char_per_block = { 5, 5, 0 },
  291. .block_w = { 4, 2, 0 }, .block_h = { 1, 1, 0 }, .hsub = 1,
  292. .vsub = 1, .is_yuv = true },
  293. { .format = DRM_FORMAT_Q410, .depth = 0,
  294. .num_planes = 3, .char_per_block = { 2, 2, 2 },
  295. .block_w = { 1, 1, 1 }, .block_h = { 1, 1, 1 }, .hsub = 1,
  296. .vsub = 1, .is_yuv = true },
  297. { .format = DRM_FORMAT_Q401, .depth = 0,
  298. .num_planes = 3, .char_per_block = { 2, 2, 2 },
  299. .block_w = { 1, 1, 1 }, .block_h = { 1, 1, 1 }, .hsub = 1,
  300. .vsub = 1, .is_yuv = true },
  301. { .format = DRM_FORMAT_P030, .depth = 0, .num_planes = 2,
  302. .char_per_block = { 4, 8, 0 }, .block_w = { 3, 3, 0 }, .block_h = { 1, 1, 0 },
  303. .hsub = 2, .vsub = 2, .is_yuv = true},
  304. };
  305. unsigned int i;
  306. for (i = 0; i < ARRAY_SIZE(formats); ++i) {
  307. if (formats[i].format == format)
  308. return &formats[i];
  309. }
  310. return NULL;
  311. }
  312. /**
  313. * drm_format_info - query information for a given format
  314. * @format: pixel format (DRM_FORMAT_*)
  315. *
  316. * The caller should only pass a supported pixel format to this function.
  317. * Unsupported pixel formats will generate a warning in the kernel log.
  318. *
  319. * Returns:
  320. * The instance of struct drm_format_info that describes the pixel format, or
  321. * NULL if the format is unsupported.
  322. */
  323. const struct drm_format_info *drm_format_info(u32 format)
  324. {
  325. const struct drm_format_info *info;
  326. info = __drm_format_info(format);
  327. WARN_ON(!info);
  328. return info;
  329. }
  330. EXPORT_SYMBOL(drm_format_info);
  331. /**
  332. * drm_get_format_info - query information for a given framebuffer configuration
  333. * @dev: DRM device
  334. * @mode_cmd: metadata from the userspace fb creation request
  335. *
  336. * Returns:
  337. * The instance of struct drm_format_info that describes the pixel format, or
  338. * NULL if the format is unsupported.
  339. */
  340. const struct drm_format_info *
  341. drm_get_format_info(struct drm_device *dev,
  342. const struct drm_mode_fb_cmd2 *mode_cmd)
  343. {
  344. const struct drm_format_info *info = NULL;
  345. if (dev->mode_config.funcs->get_format_info)
  346. info = dev->mode_config.funcs->get_format_info(mode_cmd);
  347. if (!info)
  348. info = drm_format_info(mode_cmd->pixel_format);
  349. return info;
  350. }
  351. EXPORT_SYMBOL(drm_get_format_info);
  352. /**
  353. * drm_format_info_block_width - width in pixels of block.
  354. * @info: pixel format info
  355. * @plane: plane index
  356. *
  357. * Returns:
  358. * The width in pixels of a block, depending on the plane index.
  359. */
  360. unsigned int drm_format_info_block_width(const struct drm_format_info *info,
  361. int plane)
  362. {
  363. if (!info || plane < 0 || plane >= info->num_planes)
  364. return 0;
  365. if (!info->block_w[plane])
  366. return 1;
  367. return info->block_w[plane];
  368. }
  369. EXPORT_SYMBOL(drm_format_info_block_width);
  370. /**
  371. * drm_format_info_block_height - height in pixels of a block
  372. * @info: pixel format info
  373. * @plane: plane index
  374. *
  375. * Returns:
  376. * The height in pixels of a block, depending on the plane index.
  377. */
  378. unsigned int drm_format_info_block_height(const struct drm_format_info *info,
  379. int plane)
  380. {
  381. if (!info || plane < 0 || plane >= info->num_planes)
  382. return 0;
  383. if (!info->block_h[plane])
  384. return 1;
  385. return info->block_h[plane];
  386. }
  387. EXPORT_SYMBOL(drm_format_info_block_height);
  388. /**
  389. * drm_format_info_bpp - number of bits per pixel
  390. * @info: pixel format info
  391. * @plane: plane index
  392. *
  393. * Returns:
  394. * The actual number of bits per pixel, depending on the plane index.
  395. */
  396. unsigned int drm_format_info_bpp(const struct drm_format_info *info, int plane)
  397. {
  398. if (!info || plane < 0 || plane >= info->num_planes)
  399. return 0;
  400. return info->char_per_block[plane] * 8 /
  401. (drm_format_info_block_width(info, plane) *
  402. drm_format_info_block_height(info, plane));
  403. }
  404. EXPORT_SYMBOL(drm_format_info_bpp);
  405. /**
  406. * drm_format_info_min_pitch - computes the minimum required pitch in bytes
  407. * @info: pixel format info
  408. * @plane: plane index
  409. * @buffer_width: buffer width in pixels
  410. *
  411. * Returns:
  412. * The minimum required pitch in bytes for a buffer by taking into consideration
  413. * the pixel format information and the buffer width.
  414. */
  415. uint64_t drm_format_info_min_pitch(const struct drm_format_info *info,
  416. int plane, unsigned int buffer_width)
  417. {
  418. if (!info || plane < 0 || plane >= info->num_planes)
  419. return 0;
  420. return DIV_ROUND_UP_ULL((u64)buffer_width * info->char_per_block[plane],
  421. drm_format_info_block_width(info, plane) *
  422. drm_format_info_block_height(info, plane));
  423. }
  424. EXPORT_SYMBOL(drm_format_info_min_pitch);