drm_panel_orientation_quirks.c 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207
  1. /* SPDX-License-Identifier: MIT */
  2. /*
  3. * drm_panel_orientation_quirks.c -- Quirks for non-normal panel orientation
  4. *
  5. * Copyright (C) 2017 Hans de Goede <hdegoede@redhat.com>
  6. *
  7. * Note the quirks in this file are shared with fbdev/efifb and as such
  8. * must not depend on other drm code.
  9. */
  10. #include <linux/dmi.h>
  11. #include <linux/module.h>
  12. #include <drm/drm_connector.h>
  13. #include <drm/drm_utils.h>
  14. #ifdef CONFIG_DMI
  15. /*
  16. * Some x86 clamshell design devices use portrait tablet screens and a display
  17. * engine which cannot rotate in hardware, so we need to rotate the fbcon to
  18. * compensate. Unfortunately these (cheap) devices also typically have quite
  19. * generic DMI data, so we match on a combination of DMI data, screen resolution
  20. * and a list of known BIOS dates to avoid false positives.
  21. */
  22. struct drm_dmi_panel_orientation_data {
  23. int width;
  24. int height;
  25. const char * const *bios_dates;
  26. int orientation;
  27. };
  28. static const struct drm_dmi_panel_orientation_data acer_s1003 = {
  29. .width = 800,
  30. .height = 1280,
  31. .orientation = DRM_MODE_PANEL_ORIENTATION_RIGHT_UP,
  32. };
  33. static const struct drm_dmi_panel_orientation_data asus_t100ha = {
  34. .width = 800,
  35. .height = 1280,
  36. .orientation = DRM_MODE_PANEL_ORIENTATION_LEFT_UP,
  37. };
  38. static const struct drm_dmi_panel_orientation_data gpd_pocket = {
  39. .width = 1200,
  40. .height = 1920,
  41. .bios_dates = (const char * const []){ "05/26/2017", "06/28/2017",
  42. "07/05/2017", "08/07/2017", NULL },
  43. .orientation = DRM_MODE_PANEL_ORIENTATION_RIGHT_UP,
  44. };
  45. static const struct drm_dmi_panel_orientation_data gpd_win = {
  46. .width = 720,
  47. .height = 1280,
  48. .bios_dates = (const char * const []){
  49. "10/25/2016", "11/18/2016", "12/23/2016", "12/26/2016",
  50. "02/21/2017", "03/20/2017", "05/25/2017", NULL },
  51. .orientation = DRM_MODE_PANEL_ORIENTATION_RIGHT_UP,
  52. };
  53. static const struct drm_dmi_panel_orientation_data itworks_tw891 = {
  54. .width = 800,
  55. .height = 1280,
  56. .bios_dates = (const char * const []){ "10/16/2015", NULL },
  57. .orientation = DRM_MODE_PANEL_ORIENTATION_RIGHT_UP,
  58. };
  59. static const struct drm_dmi_panel_orientation_data lcd800x1280_rightside_up = {
  60. .width = 800,
  61. .height = 1280,
  62. .orientation = DRM_MODE_PANEL_ORIENTATION_RIGHT_UP,
  63. };
  64. static const struct dmi_system_id orientation_data[] = {
  65. { /* Acer One 10 (S1003) */
  66. .matches = {
  67. DMI_EXACT_MATCH(DMI_SYS_VENDOR, "Acer"),
  68. DMI_EXACT_MATCH(DMI_PRODUCT_NAME, "One S1003"),
  69. },
  70. .driver_data = (void *)&acer_s1003,
  71. }, { /* Asus T100HA */
  72. .matches = {
  73. DMI_EXACT_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
  74. DMI_EXACT_MATCH(DMI_PRODUCT_NAME, "T100HAN"),
  75. },
  76. .driver_data = (void *)&asus_t100ha,
  77. }, { /*
  78. * GPD Pocket, note that the the DMI data is less generic then
  79. * it seems, devices with a board-vendor of "AMI Corporation"
  80. * are quite rare, as are devices which have both board- *and*
  81. * product-id set to "Default String"
  82. */
  83. .matches = {
  84. DMI_EXACT_MATCH(DMI_BOARD_VENDOR, "AMI Corporation"),
  85. DMI_EXACT_MATCH(DMI_BOARD_NAME, "Default string"),
  86. DMI_EXACT_MATCH(DMI_BOARD_SERIAL, "Default string"),
  87. DMI_EXACT_MATCH(DMI_PRODUCT_NAME, "Default string"),
  88. },
  89. .driver_data = (void *)&gpd_pocket,
  90. }, { /* GPD Win (same note on DMI match as GPD Pocket) */
  91. .matches = {
  92. DMI_EXACT_MATCH(DMI_BOARD_VENDOR, "AMI Corporation"),
  93. DMI_EXACT_MATCH(DMI_BOARD_NAME, "Default string"),
  94. DMI_EXACT_MATCH(DMI_BOARD_SERIAL, "Default string"),
  95. DMI_EXACT_MATCH(DMI_PRODUCT_NAME, "Default string"),
  96. },
  97. .driver_data = (void *)&gpd_win,
  98. }, { /* I.T.Works TW891 */
  99. .matches = {
  100. DMI_EXACT_MATCH(DMI_SYS_VENDOR, "To be filled by O.E.M."),
  101. DMI_EXACT_MATCH(DMI_PRODUCT_NAME, "TW891"),
  102. DMI_EXACT_MATCH(DMI_BOARD_VENDOR, "To be filled by O.E.M."),
  103. DMI_EXACT_MATCH(DMI_BOARD_NAME, "TW891"),
  104. },
  105. .driver_data = (void *)&itworks_tw891,
  106. }, { /*
  107. * Lenovo Ideapad Miix 310 laptop, only some production batches
  108. * have a portrait screen, the resolution checks makes the quirk
  109. * apply only to those batches.
  110. */
  111. .matches = {
  112. DMI_EXACT_MATCH(DMI_SYS_VENDOR, "LENOVO"),
  113. DMI_EXACT_MATCH(DMI_PRODUCT_NAME, "80SG"),
  114. DMI_EXACT_MATCH(DMI_PRODUCT_VERSION, "MIIX 310-10ICR"),
  115. },
  116. .driver_data = (void *)&lcd800x1280_rightside_up,
  117. }, { /* Lenovo Ideapad Miix 320 */
  118. .matches = {
  119. DMI_EXACT_MATCH(DMI_SYS_VENDOR, "LENOVO"),
  120. DMI_EXACT_MATCH(DMI_PRODUCT_NAME, "80XF"),
  121. DMI_EXACT_MATCH(DMI_PRODUCT_VERSION, "Lenovo MIIX 320-10ICR"),
  122. },
  123. .driver_data = (void *)&lcd800x1280_rightside_up,
  124. }, { /* VIOS LTH17 */
  125. .matches = {
  126. DMI_EXACT_MATCH(DMI_SYS_VENDOR, "VIOS"),
  127. DMI_EXACT_MATCH(DMI_PRODUCT_NAME, "LTH17"),
  128. },
  129. .driver_data = (void *)&lcd800x1280_rightside_up,
  130. },
  131. {}
  132. };
  133. /**
  134. * drm_get_panel_orientation_quirk - Check for panel orientation quirks
  135. * @width: width in pixels of the panel
  136. * @height: height in pixels of the panel
  137. *
  138. * This function checks for platform specific (e.g. DMI based) quirks
  139. * providing info on panel_orientation for systems where this cannot be
  140. * probed from the hard-/firm-ware. To avoid false-positive this function
  141. * takes the panel resolution as argument and checks that against the
  142. * resolution expected by the quirk-table entry.
  143. *
  144. * Note this function is also used outside of the drm-subsys, by for example
  145. * the efifb code. Because of this this function gets compiled into its own
  146. * kernel-module when built as a module.
  147. *
  148. * Returns:
  149. * A DRM_MODE_PANEL_ORIENTATION_* value if there is a quirk for this system,
  150. * or DRM_MODE_PANEL_ORIENTATION_UNKNOWN if there is no quirk.
  151. */
  152. int drm_get_panel_orientation_quirk(int width, int height)
  153. {
  154. const struct dmi_system_id *match;
  155. const struct drm_dmi_panel_orientation_data *data;
  156. const char *bios_date;
  157. int i;
  158. for (match = dmi_first_match(orientation_data);
  159. match;
  160. match = dmi_first_match(match + 1)) {
  161. data = match->driver_data;
  162. if (data->width != width ||
  163. data->height != height)
  164. continue;
  165. if (!data->bios_dates)
  166. return data->orientation;
  167. bios_date = dmi_get_system_info(DMI_BIOS_DATE);
  168. if (!bios_date)
  169. continue;
  170. i = match_string(data->bios_dates, -1, bios_date);
  171. if (i >= 0)
  172. return data->orientation;
  173. }
  174. return DRM_MODE_PANEL_ORIENTATION_UNKNOWN;
  175. }
  176. EXPORT_SYMBOL(drm_get_panel_orientation_quirk);
  177. #else
  178. /* There are no quirks for non x86 devices yet */
  179. int drm_get_panel_orientation_quirk(int width, int height)
  180. {
  181. return DRM_MODE_PANEL_ORIENTATION_UNKNOWN;
  182. }
  183. EXPORT_SYMBOL(drm_get_panel_orientation_quirk);
  184. #endif
  185. MODULE_LICENSE("Dual MIT/GPL");