gud_connector.c 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725
  1. // SPDX-License-Identifier: MIT
  2. /*
  3. * Copyright 2020 Noralf Trønnes
  4. */
  5. #include <linux/backlight.h>
  6. #include <linux/workqueue.h>
  7. #include <drm/drm_atomic.h>
  8. #include <drm/drm_atomic_state_helper.h>
  9. #include <drm/drm_connector.h>
  10. #include <drm/drm_drv.h>
  11. #include <drm/drm_edid.h>
  12. #include <drm/drm_encoder.h>
  13. #include <drm/drm_file.h>
  14. #include <drm/drm_modeset_helper_vtables.h>
  15. #include <drm/drm_print.h>
  16. #include <drm/drm_probe_helper.h>
  17. #include <drm/gud.h>
  18. #include "gud_internal.h"
  19. struct gud_connector {
  20. struct drm_connector connector;
  21. struct drm_encoder encoder;
  22. struct backlight_device *backlight;
  23. struct work_struct backlight_work;
  24. /* Supported properties */
  25. u16 *properties;
  26. unsigned int num_properties;
  27. /* Initial gadget tv state if applicable, applied on state reset */
  28. struct drm_tv_connector_state initial_tv_state;
  29. /*
  30. * Initial gadget backlight brightness if applicable, applied on state reset.
  31. * The value -ENODEV is used to signal no backlight.
  32. */
  33. int initial_brightness;
  34. };
  35. static inline struct gud_connector *to_gud_connector(struct drm_connector *connector)
  36. {
  37. return container_of(connector, struct gud_connector, connector);
  38. }
  39. static void gud_conn_err(struct drm_connector *connector, const char *msg, int ret)
  40. {
  41. dev_err(connector->dev->dev, "%s: %s (ret=%d)\n", connector->name, msg, ret);
  42. }
  43. /*
  44. * Use a worker to avoid taking kms locks inside the backlight lock.
  45. * Other display drivers use backlight within their kms locks.
  46. * This avoids inconsistent locking rules, which would upset lockdep.
  47. */
  48. static void gud_connector_backlight_update_status_work(struct work_struct *work)
  49. {
  50. struct gud_connector *gconn = container_of(work, struct gud_connector, backlight_work);
  51. struct drm_connector *connector = &gconn->connector;
  52. struct drm_connector_state *connector_state;
  53. struct drm_device *drm = connector->dev;
  54. struct drm_modeset_acquire_ctx ctx;
  55. struct drm_atomic_state *state;
  56. int idx, ret;
  57. if (!drm_dev_enter(drm, &idx))
  58. return;
  59. state = drm_atomic_state_alloc(drm);
  60. if (!state) {
  61. ret = -ENOMEM;
  62. goto exit;
  63. }
  64. drm_modeset_acquire_init(&ctx, 0);
  65. state->acquire_ctx = &ctx;
  66. retry:
  67. connector_state = drm_atomic_get_connector_state(state, connector);
  68. if (IS_ERR(connector_state)) {
  69. ret = PTR_ERR(connector_state);
  70. goto out;
  71. }
  72. /* Reuse tv.brightness to avoid having to subclass */
  73. connector_state->tv.brightness = gconn->backlight->props.brightness;
  74. ret = drm_atomic_commit(state);
  75. out:
  76. if (ret == -EDEADLK) {
  77. drm_atomic_state_clear(state);
  78. drm_modeset_backoff(&ctx);
  79. goto retry;
  80. }
  81. drm_atomic_state_put(state);
  82. drm_modeset_drop_locks(&ctx);
  83. drm_modeset_acquire_fini(&ctx);
  84. exit:
  85. drm_dev_exit(idx);
  86. if (ret)
  87. dev_err(drm->dev, "Failed to update backlight, err=%d\n", ret);
  88. }
  89. static int gud_connector_backlight_update_status(struct backlight_device *bd)
  90. {
  91. struct drm_connector *connector = bl_get_data(bd);
  92. struct gud_connector *gconn = to_gud_connector(connector);
  93. /* The USB timeout is 5 seconds so use system_long_wq for worst case scenario */
  94. queue_work(system_long_wq, &gconn->backlight_work);
  95. return 0;
  96. }
  97. static const struct backlight_ops gud_connector_backlight_ops = {
  98. .update_status = gud_connector_backlight_update_status,
  99. };
  100. static int gud_connector_backlight_register(struct gud_connector *gconn)
  101. {
  102. struct drm_connector *connector = &gconn->connector;
  103. struct backlight_device *bd;
  104. const char *name;
  105. const struct backlight_properties props = {
  106. .type = BACKLIGHT_RAW,
  107. .scale = BACKLIGHT_SCALE_NON_LINEAR,
  108. .max_brightness = 100,
  109. .brightness = gconn->initial_brightness,
  110. };
  111. name = kasprintf(GFP_KERNEL, "card%d-%s-backlight",
  112. connector->dev->primary->index, connector->name);
  113. if (!name)
  114. return -ENOMEM;
  115. bd = backlight_device_register(name, connector->kdev, connector,
  116. &gud_connector_backlight_ops, &props);
  117. kfree(name);
  118. if (IS_ERR(bd))
  119. return PTR_ERR(bd);
  120. gconn->backlight = bd;
  121. return 0;
  122. }
  123. static int gud_connector_detect(struct drm_connector *connector,
  124. struct drm_modeset_acquire_ctx *ctx, bool force)
  125. {
  126. struct gud_device *gdrm = to_gud_device(connector->dev);
  127. int idx, ret;
  128. u8 status;
  129. if (!drm_dev_enter(connector->dev, &idx))
  130. return connector_status_disconnected;
  131. if (force) {
  132. ret = gud_usb_set(gdrm, GUD_REQ_SET_CONNECTOR_FORCE_DETECT,
  133. connector->index, NULL, 0);
  134. if (ret) {
  135. ret = connector_status_unknown;
  136. goto exit;
  137. }
  138. }
  139. ret = gud_usb_get_u8(gdrm, GUD_REQ_GET_CONNECTOR_STATUS, connector->index, &status);
  140. if (ret) {
  141. ret = connector_status_unknown;
  142. goto exit;
  143. }
  144. switch (status & GUD_CONNECTOR_STATUS_CONNECTED_MASK) {
  145. case GUD_CONNECTOR_STATUS_DISCONNECTED:
  146. ret = connector_status_disconnected;
  147. break;
  148. case GUD_CONNECTOR_STATUS_CONNECTED:
  149. ret = connector_status_connected;
  150. break;
  151. default:
  152. ret = connector_status_unknown;
  153. break;
  154. }
  155. if (status & GUD_CONNECTOR_STATUS_CHANGED)
  156. connector->epoch_counter += 1;
  157. exit:
  158. drm_dev_exit(idx);
  159. return ret;
  160. }
  161. struct gud_connector_get_edid_ctx {
  162. void *buf;
  163. size_t len;
  164. bool edid_override;
  165. };
  166. static int gud_connector_get_edid_block(void *data, u8 *buf, unsigned int block, size_t len)
  167. {
  168. struct gud_connector_get_edid_ctx *ctx = data;
  169. size_t start = block * EDID_LENGTH;
  170. ctx->edid_override = false;
  171. if (start + len > ctx->len)
  172. return -1;
  173. memcpy(buf, ctx->buf + start, len);
  174. return 0;
  175. }
  176. static int gud_connector_get_modes(struct drm_connector *connector)
  177. {
  178. struct gud_device *gdrm = to_gud_device(connector->dev);
  179. struct gud_display_mode_req *reqmodes = NULL;
  180. struct gud_connector_get_edid_ctx edid_ctx;
  181. unsigned int i, num_modes = 0;
  182. const struct drm_edid *drm_edid = NULL;
  183. int idx, ret;
  184. if (!drm_dev_enter(connector->dev, &idx))
  185. return 0;
  186. edid_ctx.edid_override = true;
  187. edid_ctx.buf = kmalloc(GUD_CONNECTOR_MAX_EDID_LEN, GFP_KERNEL);
  188. if (!edid_ctx.buf)
  189. goto out;
  190. ret = gud_usb_get(gdrm, GUD_REQ_GET_CONNECTOR_EDID, connector->index,
  191. edid_ctx.buf, GUD_CONNECTOR_MAX_EDID_LEN);
  192. if (ret > 0 && ret % EDID_LENGTH) {
  193. gud_conn_err(connector, "Invalid EDID size", ret);
  194. } else if (ret > 0) {
  195. edid_ctx.len = ret;
  196. drm_edid = drm_edid_read_custom(connector, gud_connector_get_edid_block, &edid_ctx);
  197. }
  198. kfree(edid_ctx.buf);
  199. drm_edid_connector_update(connector, drm_edid);
  200. if (drm_edid && edid_ctx.edid_override)
  201. goto out;
  202. reqmodes = kmalloc_objs(*reqmodes, GUD_CONNECTOR_MAX_NUM_MODES);
  203. if (!reqmodes)
  204. goto out;
  205. ret = gud_usb_get(gdrm, GUD_REQ_GET_CONNECTOR_MODES, connector->index,
  206. reqmodes, GUD_CONNECTOR_MAX_NUM_MODES * sizeof(*reqmodes));
  207. if (ret <= 0)
  208. goto out;
  209. if (ret % sizeof(*reqmodes)) {
  210. gud_conn_err(connector, "Invalid display mode array size", ret);
  211. goto out;
  212. }
  213. num_modes = ret / sizeof(*reqmodes);
  214. for (i = 0; i < num_modes; i++) {
  215. struct drm_display_mode *mode;
  216. mode = drm_mode_create(connector->dev);
  217. if (!mode) {
  218. num_modes = i;
  219. goto out;
  220. }
  221. gud_to_display_mode(mode, &reqmodes[i]);
  222. drm_mode_probed_add(connector, mode);
  223. }
  224. out:
  225. if (!num_modes)
  226. num_modes = drm_edid_connector_add_modes(connector);
  227. kfree(reqmodes);
  228. drm_edid_free(drm_edid);
  229. drm_dev_exit(idx);
  230. return num_modes;
  231. }
  232. static int gud_connector_atomic_check(struct drm_connector *connector,
  233. struct drm_atomic_state *state)
  234. {
  235. struct drm_connector_state *new_state;
  236. struct drm_crtc_state *new_crtc_state;
  237. struct drm_connector_state *old_state;
  238. new_state = drm_atomic_get_new_connector_state(state, connector);
  239. if (!new_state->crtc)
  240. return 0;
  241. old_state = drm_atomic_get_old_connector_state(state, connector);
  242. new_crtc_state = drm_atomic_get_new_crtc_state(state, new_state->crtc);
  243. if (old_state->tv.margins.left != new_state->tv.margins.left ||
  244. old_state->tv.margins.right != new_state->tv.margins.right ||
  245. old_state->tv.margins.top != new_state->tv.margins.top ||
  246. old_state->tv.margins.bottom != new_state->tv.margins.bottom ||
  247. old_state->tv.legacy_mode != new_state->tv.legacy_mode ||
  248. old_state->tv.brightness != new_state->tv.brightness ||
  249. old_state->tv.contrast != new_state->tv.contrast ||
  250. old_state->tv.flicker_reduction != new_state->tv.flicker_reduction ||
  251. old_state->tv.overscan != new_state->tv.overscan ||
  252. old_state->tv.saturation != new_state->tv.saturation ||
  253. old_state->tv.hue != new_state->tv.hue)
  254. new_crtc_state->connectors_changed = true;
  255. return 0;
  256. }
  257. static const struct drm_connector_helper_funcs gud_connector_helper_funcs = {
  258. .detect_ctx = gud_connector_detect,
  259. .get_modes = gud_connector_get_modes,
  260. .atomic_check = gud_connector_atomic_check,
  261. };
  262. static int gud_connector_late_register(struct drm_connector *connector)
  263. {
  264. struct gud_connector *gconn = to_gud_connector(connector);
  265. if (gconn->initial_brightness < 0)
  266. return 0;
  267. return gud_connector_backlight_register(gconn);
  268. }
  269. static void gud_connector_early_unregister(struct drm_connector *connector)
  270. {
  271. struct gud_connector *gconn = to_gud_connector(connector);
  272. backlight_device_unregister(gconn->backlight);
  273. cancel_work_sync(&gconn->backlight_work);
  274. }
  275. static void gud_connector_destroy(struct drm_connector *connector)
  276. {
  277. struct gud_connector *gconn = to_gud_connector(connector);
  278. drm_connector_cleanup(connector);
  279. kfree(gconn->properties);
  280. kfree(gconn);
  281. }
  282. static void gud_connector_reset(struct drm_connector *connector)
  283. {
  284. struct gud_connector *gconn = to_gud_connector(connector);
  285. drm_atomic_helper_connector_reset(connector);
  286. connector->state->tv = gconn->initial_tv_state;
  287. /* Set margins from command line */
  288. drm_atomic_helper_connector_tv_margins_reset(connector);
  289. if (gconn->initial_brightness >= 0)
  290. connector->state->tv.brightness = gconn->initial_brightness;
  291. }
  292. static const struct drm_connector_funcs gud_connector_funcs = {
  293. .fill_modes = drm_helper_probe_single_connector_modes,
  294. .late_register = gud_connector_late_register,
  295. .early_unregister = gud_connector_early_unregister,
  296. .destroy = gud_connector_destroy,
  297. .reset = gud_connector_reset,
  298. .atomic_duplicate_state = drm_atomic_helper_connector_duplicate_state,
  299. .atomic_destroy_state = drm_atomic_helper_connector_destroy_state,
  300. };
  301. /*
  302. * The tv.mode property is shared among the connectors and its enum names are
  303. * driver specific. This means that if more than one connector uses tv.mode,
  304. * the enum names has to be the same.
  305. */
  306. static int gud_connector_add_tv_mode(struct gud_device *gdrm, struct drm_connector *connector)
  307. {
  308. size_t buf_len = GUD_CONNECTOR_TV_MODE_MAX_NUM * GUD_CONNECTOR_TV_MODE_NAME_LEN;
  309. const char *modes[GUD_CONNECTOR_TV_MODE_MAX_NUM];
  310. unsigned int i, num_modes;
  311. char *buf;
  312. int ret;
  313. buf = kmalloc(buf_len, GFP_KERNEL);
  314. if (!buf)
  315. return -ENOMEM;
  316. ret = gud_usb_get(gdrm, GUD_REQ_GET_CONNECTOR_TV_MODE_VALUES,
  317. connector->index, buf, buf_len);
  318. if (ret < 0)
  319. goto free;
  320. if (!ret || ret % GUD_CONNECTOR_TV_MODE_NAME_LEN) {
  321. ret = -EIO;
  322. goto free;
  323. }
  324. num_modes = ret / GUD_CONNECTOR_TV_MODE_NAME_LEN;
  325. for (i = 0; i < num_modes; i++)
  326. modes[i] = &buf[i * GUD_CONNECTOR_TV_MODE_NAME_LEN];
  327. ret = drm_mode_create_tv_properties_legacy(connector->dev, num_modes, modes);
  328. free:
  329. kfree(buf);
  330. if (ret < 0)
  331. gud_conn_err(connector, "Failed to add TV modes", ret);
  332. return ret;
  333. }
  334. static struct drm_property *
  335. gud_connector_property_lookup(struct drm_connector *connector, u16 prop)
  336. {
  337. struct drm_mode_config *config = &connector->dev->mode_config;
  338. switch (prop) {
  339. case GUD_PROPERTY_TV_LEFT_MARGIN:
  340. return config->tv_left_margin_property;
  341. case GUD_PROPERTY_TV_RIGHT_MARGIN:
  342. return config->tv_right_margin_property;
  343. case GUD_PROPERTY_TV_TOP_MARGIN:
  344. return config->tv_top_margin_property;
  345. case GUD_PROPERTY_TV_BOTTOM_MARGIN:
  346. return config->tv_bottom_margin_property;
  347. case GUD_PROPERTY_TV_MODE:
  348. return config->legacy_tv_mode_property;
  349. case GUD_PROPERTY_TV_BRIGHTNESS:
  350. return config->tv_brightness_property;
  351. case GUD_PROPERTY_TV_CONTRAST:
  352. return config->tv_contrast_property;
  353. case GUD_PROPERTY_TV_FLICKER_REDUCTION:
  354. return config->tv_flicker_reduction_property;
  355. case GUD_PROPERTY_TV_OVERSCAN:
  356. return config->tv_overscan_property;
  357. case GUD_PROPERTY_TV_SATURATION:
  358. return config->tv_saturation_property;
  359. case GUD_PROPERTY_TV_HUE:
  360. return config->tv_hue_property;
  361. default:
  362. return ERR_PTR(-EINVAL);
  363. }
  364. }
  365. static unsigned int *gud_connector_tv_state_val(u16 prop, struct drm_tv_connector_state *state)
  366. {
  367. switch (prop) {
  368. case GUD_PROPERTY_TV_LEFT_MARGIN:
  369. return &state->margins.left;
  370. case GUD_PROPERTY_TV_RIGHT_MARGIN:
  371. return &state->margins.right;
  372. case GUD_PROPERTY_TV_TOP_MARGIN:
  373. return &state->margins.top;
  374. case GUD_PROPERTY_TV_BOTTOM_MARGIN:
  375. return &state->margins.bottom;
  376. case GUD_PROPERTY_TV_MODE:
  377. return &state->legacy_mode;
  378. case GUD_PROPERTY_TV_BRIGHTNESS:
  379. return &state->brightness;
  380. case GUD_PROPERTY_TV_CONTRAST:
  381. return &state->contrast;
  382. case GUD_PROPERTY_TV_FLICKER_REDUCTION:
  383. return &state->flicker_reduction;
  384. case GUD_PROPERTY_TV_OVERSCAN:
  385. return &state->overscan;
  386. case GUD_PROPERTY_TV_SATURATION:
  387. return &state->saturation;
  388. case GUD_PROPERTY_TV_HUE:
  389. return &state->hue;
  390. default:
  391. return ERR_PTR(-EINVAL);
  392. }
  393. }
  394. static int gud_connector_add_properties(struct gud_device *gdrm, struct gud_connector *gconn)
  395. {
  396. struct drm_connector *connector = &gconn->connector;
  397. struct drm_device *drm = &gdrm->drm;
  398. struct gud_property_req *properties;
  399. unsigned int i, num_properties;
  400. int ret;
  401. properties = kzalloc_objs(*properties, GUD_CONNECTOR_PROPERTIES_MAX_NUM);
  402. if (!properties)
  403. return -ENOMEM;
  404. ret = gud_usb_get(gdrm, GUD_REQ_GET_CONNECTOR_PROPERTIES, connector->index,
  405. properties, GUD_CONNECTOR_PROPERTIES_MAX_NUM * sizeof(*properties));
  406. if (ret <= 0)
  407. goto out;
  408. if (ret % sizeof(*properties)) {
  409. ret = -EIO;
  410. goto out;
  411. }
  412. num_properties = ret / sizeof(*properties);
  413. ret = 0;
  414. gconn->properties = kcalloc(num_properties, sizeof(*gconn->properties), GFP_KERNEL);
  415. if (!gconn->properties) {
  416. ret = -ENOMEM;
  417. goto out;
  418. }
  419. for (i = 0; i < num_properties; i++) {
  420. u16 prop = le16_to_cpu(properties[i].prop);
  421. u64 val = le64_to_cpu(properties[i].val);
  422. struct drm_property *property;
  423. unsigned int *state_val;
  424. drm_dbg(drm, "property: %u = %llu(0x%llx)\n", prop, val, val);
  425. switch (prop) {
  426. case GUD_PROPERTY_TV_LEFT_MARGIN:
  427. fallthrough;
  428. case GUD_PROPERTY_TV_RIGHT_MARGIN:
  429. fallthrough;
  430. case GUD_PROPERTY_TV_TOP_MARGIN:
  431. fallthrough;
  432. case GUD_PROPERTY_TV_BOTTOM_MARGIN:
  433. ret = drm_mode_create_tv_margin_properties(drm);
  434. if (ret)
  435. goto out;
  436. break;
  437. case GUD_PROPERTY_TV_MODE:
  438. ret = gud_connector_add_tv_mode(gdrm, connector);
  439. if (ret)
  440. goto out;
  441. break;
  442. case GUD_PROPERTY_TV_BRIGHTNESS:
  443. fallthrough;
  444. case GUD_PROPERTY_TV_CONTRAST:
  445. fallthrough;
  446. case GUD_PROPERTY_TV_FLICKER_REDUCTION:
  447. fallthrough;
  448. case GUD_PROPERTY_TV_OVERSCAN:
  449. fallthrough;
  450. case GUD_PROPERTY_TV_SATURATION:
  451. fallthrough;
  452. case GUD_PROPERTY_TV_HUE:
  453. /* This is a no-op if already added. */
  454. ret = drm_mode_create_tv_properties_legacy(drm, 0, NULL);
  455. if (ret)
  456. goto out;
  457. break;
  458. case GUD_PROPERTY_BACKLIGHT_BRIGHTNESS:
  459. if (val > 100) {
  460. ret = -EINVAL;
  461. goto out;
  462. }
  463. gconn->initial_brightness = val;
  464. break;
  465. default:
  466. /* New ones might show up in future devices, skip those we don't know. */
  467. drm_dbg(drm, "Ignoring unknown property: %u\n", prop);
  468. continue;
  469. }
  470. gconn->properties[gconn->num_properties++] = prop;
  471. if (prop == GUD_PROPERTY_BACKLIGHT_BRIGHTNESS)
  472. continue; /* not a DRM property */
  473. property = gud_connector_property_lookup(connector, prop);
  474. if (drm_WARN_ON(drm, IS_ERR(property)))
  475. continue;
  476. state_val = gud_connector_tv_state_val(prop, &gconn->initial_tv_state);
  477. if (drm_WARN_ON(drm, IS_ERR(state_val)))
  478. continue;
  479. *state_val = val;
  480. drm_object_attach_property(&connector->base, property, 0);
  481. }
  482. out:
  483. kfree(properties);
  484. return ret;
  485. }
  486. int gud_connector_fill_properties(struct drm_connector_state *connector_state,
  487. struct gud_property_req *properties)
  488. {
  489. struct gud_connector *gconn = to_gud_connector(connector_state->connector);
  490. unsigned int i;
  491. for (i = 0; i < gconn->num_properties; i++) {
  492. u16 prop = gconn->properties[i];
  493. u64 val;
  494. if (prop == GUD_PROPERTY_BACKLIGHT_BRIGHTNESS) {
  495. val = connector_state->tv.brightness;
  496. } else {
  497. unsigned int *state_val;
  498. state_val = gud_connector_tv_state_val(prop, &connector_state->tv);
  499. if (drm_WARN_ON_ONCE(connector_state->connector->dev, IS_ERR(state_val)))
  500. return PTR_ERR(state_val);
  501. val = *state_val;
  502. }
  503. properties[i].prop = cpu_to_le16(prop);
  504. properties[i].val = cpu_to_le64(val);
  505. }
  506. return gconn->num_properties;
  507. }
  508. static const struct drm_encoder_funcs gud_drm_simple_encoder_funcs_cleanup = {
  509. .destroy = drm_encoder_cleanup,
  510. };
  511. static int gud_connector_create(struct gud_device *gdrm, unsigned int index,
  512. struct gud_connector_descriptor_req *desc)
  513. {
  514. struct drm_device *drm = &gdrm->drm;
  515. struct gud_connector *gconn;
  516. struct drm_connector *connector;
  517. int ret, connector_type;
  518. u32 flags;
  519. gconn = kzalloc_obj(*gconn);
  520. if (!gconn)
  521. return -ENOMEM;
  522. INIT_WORK(&gconn->backlight_work, gud_connector_backlight_update_status_work);
  523. gconn->initial_brightness = -ENODEV;
  524. flags = le32_to_cpu(desc->flags);
  525. connector = &gconn->connector;
  526. drm_dbg(drm, "Connector: index=%u type=%u flags=0x%x\n", index, desc->connector_type, flags);
  527. switch (desc->connector_type) {
  528. case GUD_CONNECTOR_TYPE_PANEL:
  529. connector_type = DRM_MODE_CONNECTOR_USB;
  530. break;
  531. case GUD_CONNECTOR_TYPE_VGA:
  532. connector_type = DRM_MODE_CONNECTOR_VGA;
  533. break;
  534. case GUD_CONNECTOR_TYPE_DVI:
  535. connector_type = DRM_MODE_CONNECTOR_DVID;
  536. break;
  537. case GUD_CONNECTOR_TYPE_COMPOSITE:
  538. connector_type = DRM_MODE_CONNECTOR_Composite;
  539. break;
  540. case GUD_CONNECTOR_TYPE_SVIDEO:
  541. connector_type = DRM_MODE_CONNECTOR_SVIDEO;
  542. break;
  543. case GUD_CONNECTOR_TYPE_COMPONENT:
  544. connector_type = DRM_MODE_CONNECTOR_Component;
  545. break;
  546. case GUD_CONNECTOR_TYPE_DISPLAYPORT:
  547. connector_type = DRM_MODE_CONNECTOR_DisplayPort;
  548. break;
  549. case GUD_CONNECTOR_TYPE_HDMI:
  550. connector_type = DRM_MODE_CONNECTOR_HDMIA;
  551. break;
  552. default: /* future types */
  553. connector_type = DRM_MODE_CONNECTOR_USB;
  554. break;
  555. }
  556. drm_connector_helper_add(connector, &gud_connector_helper_funcs);
  557. ret = drm_connector_init(drm, connector, &gud_connector_funcs, connector_type);
  558. if (ret) {
  559. kfree(connector);
  560. return ret;
  561. }
  562. if (drm_WARN_ON(drm, connector->index != index))
  563. return -EINVAL;
  564. if (flags & GUD_CONNECTOR_FLAGS_POLL_STATUS)
  565. connector->polled = (DRM_CONNECTOR_POLL_CONNECT | DRM_CONNECTOR_POLL_DISCONNECT);
  566. if (flags & GUD_CONNECTOR_FLAGS_INTERLACE)
  567. connector->interlace_allowed = true;
  568. if (flags & GUD_CONNECTOR_FLAGS_DOUBLESCAN)
  569. connector->doublescan_allowed = true;
  570. ret = gud_connector_add_properties(gdrm, gconn);
  571. if (ret) {
  572. gud_conn_err(connector, "Failed to add properties", ret);
  573. return ret;
  574. }
  575. gconn->encoder.possible_crtcs = drm_crtc_mask(&gdrm->crtc);
  576. ret = drm_encoder_init(drm, &gconn->encoder, &gud_drm_simple_encoder_funcs_cleanup,
  577. DRM_MODE_ENCODER_NONE, NULL);
  578. if (ret)
  579. return ret;
  580. return drm_connector_attach_encoder(connector, &gconn->encoder);
  581. }
  582. int gud_get_connectors(struct gud_device *gdrm)
  583. {
  584. struct gud_connector_descriptor_req *descs;
  585. unsigned int i, num_connectors;
  586. int ret;
  587. descs = kmalloc_objs(*descs, GUD_CONNECTORS_MAX_NUM);
  588. if (!descs)
  589. return -ENOMEM;
  590. ret = gud_usb_get(gdrm, GUD_REQ_GET_CONNECTORS, 0,
  591. descs, GUD_CONNECTORS_MAX_NUM * sizeof(*descs));
  592. if (ret < 0)
  593. goto free;
  594. if (!ret || ret % sizeof(*descs)) {
  595. ret = -EIO;
  596. goto free;
  597. }
  598. num_connectors = ret / sizeof(*descs);
  599. for (i = 0; i < num_connectors; i++) {
  600. ret = gud_connector_create(gdrm, i, &descs[i]);
  601. if (ret)
  602. goto free;
  603. }
  604. free:
  605. kfree(descs);
  606. return ret;
  607. }