graphic.c 38 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495
  1. /*
  2. * Copyright (c) 2006-2023, RT-Thread Development Team
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. *
  6. * Change Logs:
  7. * Date Author Notes
  8. * 2023-02-25 GuEe-GUI the first version
  9. */
  10. #include <rthw.h>
  11. #include <rtthread.h>
  12. #include <rtdevice.h>
  13. #define DBG_TAG "rtdm.graphic"
  14. #define DBG_LVL DBG_INFO
  15. #include <rtdbg.h>
  16. #define raw_to_graphic(dev) rt_container_of(dev, struct rt_graphic_device, parent)
  17. struct fb_format
  18. {
  19. rt_uint32_t mode;
  20. rt_uint32_t bits_per_pixel;
  21. struct fb_bitfield red;
  22. struct fb_bitfield green;
  23. struct fb_bitfield blue;
  24. struct fb_bitfield transp;
  25. };
  26. static const struct fb_format graphic_formats[] =
  27. {
  28. { RTGRAPHIC_PIXEL_FORMAT_GRAY4, 4, },
  29. { RTGRAPHIC_PIXEL_FORMAT_GRAY16, 16, },
  30. { RTGRAPHIC_PIXEL_FORMAT_RGB332, 8, { 5, 3 }, { 2, 3 }, { 0, 2 }, },
  31. { RTGRAPHIC_PIXEL_FORMAT_RGB444, 12, { 8, 4 }, { 4, 4 }, { 0, 4 }, },
  32. { RTGRAPHIC_PIXEL_FORMAT_RGB565, 16, { 11, 5 }, { 5, 6 }, { 0, 5 }, },
  33. { RTGRAPHIC_PIXEL_FORMAT_RGB565P, 16, { 0, 5 }, { 5, 6 }, { 11, 5 }, },
  34. { RTGRAPHIC_PIXEL_FORMAT_BGR565, 16, { 0, 5 }, { 5, 6 }, { 11, 5 }, },
  35. { RTGRAPHIC_PIXEL_FORMAT_RGB666, 18, { 12, 6 }, { 6, 6 }, { 0, 6 }, },
  36. { RTGRAPHIC_PIXEL_FORMAT_RGB888, 24, { 16, 8 }, { 8, 8 }, { 0, 8 }, },
  37. { RTGRAPHIC_PIXEL_FORMAT_BGR888, 24, { 0, 8 }, { 8, 8 }, { 16, 8 }, },
  38. { RTGRAPHIC_PIXEL_FORMAT_ARGB888, 32, { 16, 8 }, { 8, 8 }, { 0, 8 }, { 24, 8 }, },
  39. { RTGRAPHIC_PIXEL_FORMAT_ABGR888, 32, { 0, 8 }, { 8, 8 }, { 16, 8 }, { 24, 8 }, },
  40. };
  41. /* RT-Thread device max id is 255 */
  42. static rt_uint8_t fbcon_map[256] = {};
  43. static struct rt_dm_ida graphic_ida = RT_DM_IDA_INIT(GRAPHIC_FRAMEBUFFER);
  44. rt_inline void spin_lock(struct rt_spinlock *spinlock)
  45. {
  46. rt_hw_spin_lock(&spinlock->lock);
  47. }
  48. rt_inline void spin_unlock(struct rt_spinlock *spinlock)
  49. {
  50. rt_hw_spin_unlock(&spinlock->lock);
  51. }
  52. static rt_bool_t plane_need_update(struct rt_graphic_plane *plane)
  53. {
  54. if (plane->ops->update)
  55. {
  56. if (!plane->graphic->update_timer)
  57. {
  58. return RT_TRUE;
  59. }
  60. }
  61. return RT_FALSE;
  62. }
  63. static struct rt_graphic_plane *plane_get_current(struct rt_graphic_device *gdev)
  64. {
  65. if (gdev->ops->current_plane)
  66. {
  67. return gdev->ops->current_plane(gdev);
  68. }
  69. return gdev->primary_plane;
  70. }
  71. static rt_err_t plane_fb_remap(struct rt_graphic_plane *plane,
  72. rt_uint32_t mode, struct rt_device_rect_info *rect)
  73. {
  74. rt_err_t err;
  75. if (plane->ops->fb_cleanup && (err = plane->ops->fb_cleanup(plane)))
  76. {
  77. return err;
  78. }
  79. if (!(err = plane->ops->fb_remap(plane, mode, rect)))
  80. {
  81. plane->mode = mode;
  82. plane->x = rect->x;
  83. plane->y = rect->y;
  84. plane->width = rect->width;
  85. plane->height = rect->height;
  86. }
  87. return err;
  88. }
  89. static rt_err_t plane_fb_pan_display(struct rt_graphic_plane *plane,
  90. struct rt_device_rect_info *rect)
  91. {
  92. void *framebuffer_end = plane->framebuffer;
  93. rt_size_t byte_per_pixel = plane->bits_per_pixel / 8;
  94. framebuffer_end += rect->x * byte_per_pixel;
  95. framebuffer_end += rect->y * plane->line_length;
  96. framebuffer_end += rect->width * rect->height * byte_per_pixel;
  97. if (framebuffer_end < plane->framebuffer + plane->framebuffer_len)
  98. {
  99. return plane->ops->fb_pan_display(plane, rect);
  100. }
  101. return -RT_EINVAL;
  102. }
  103. static rt_err_t graphic_dpms_switch(struct rt_graphic_device *gdev, rt_uint32_t dpms)
  104. {
  105. rt_err_t err;
  106. if (!(err = gdev->ops->dpms_switch(gdev, dpms)))
  107. {
  108. gdev->dpms = dpms;
  109. }
  110. return err;
  111. }
  112. static rt_err_t _graphic_open(rt_device_t dev, rt_uint16_t oflag)
  113. {
  114. struct rt_graphic_device *gdev = raw_to_graphic(dev);
  115. struct rt_graphic_plane *plane = gdev->primary_plane;
  116. if (dev->ref_count > 0 && (oflag & RT_DEVICE_OFLAG_WRONLY))
  117. {
  118. return -RT_EBUSY;
  119. }
  120. if (plane->ops->fb_pan_display)
  121. {
  122. struct rt_device_rect_info rect;
  123. rect.x = 0;
  124. rect.y = 0;
  125. rect.width = plane->width;
  126. rect.height = plane->height;
  127. return plane->ops->fb_pan_display(plane, &rect);
  128. }
  129. return RT_EOK;
  130. }
  131. static rt_ssize_t _graphic_read(rt_device_t dev, rt_off_t pos, void *buffer, rt_size_t size)
  132. {
  133. rt_ssize_t res;
  134. struct rt_graphic_device *gdev = raw_to_graphic(dev);
  135. struct rt_graphic_plane *plane = gdev->primary_plane;
  136. res = rt_min_t(rt_ssize_t, plane->framebuffer_len - pos, size);
  137. if (res > 0)
  138. {
  139. rt_memcpy(buffer, plane->framebuffer + pos, res);
  140. }
  141. else
  142. {
  143. res = 0;
  144. }
  145. return res;
  146. }
  147. static rt_ssize_t _graphic_write(rt_device_t dev, rt_off_t pos, const void *buffer, rt_size_t size)
  148. {
  149. rt_ssize_t res;
  150. struct rt_graphic_device *gdev = raw_to_graphic(dev);
  151. struct rt_graphic_plane *plane = gdev->primary_plane;
  152. res = rt_min_t(rt_ssize_t, plane->framebuffer_len - pos, size);
  153. if (res > 0)
  154. {
  155. rt_memcpy(plane->framebuffer + pos, buffer, res);
  156. }
  157. else
  158. {
  159. res = 0;
  160. }
  161. return res;
  162. }
  163. static rt_err_t _graphic_control(rt_device_t dev, int cmd, void *args)
  164. {
  165. rt_err_t err = RT_EOK;
  166. rt_bool_t need_schedule = RT_FALSE;
  167. struct rt_graphic_device *gdev = raw_to_graphic(dev);
  168. _retry:
  169. if (need_schedule)
  170. {
  171. rt_thread_yield();
  172. }
  173. spin_lock(&gdev->lock);
  174. switch (cmd)
  175. {
  176. case RT_DEVICE_CTRL_CURSOR_SET_POSITION:
  177. if (args)
  178. {
  179. struct rt_graphic_plane *plane = gdev->cursor_plane;
  180. if (plane)
  181. {
  182. rt_uint32_t position = (rt_uint32_t)(rt_ubase_t)args;
  183. plane->x = position >> 16;
  184. plane->y = position & 0xffff;
  185. if (plane_need_update(plane))
  186. {
  187. struct rt_device_rect_info rect;
  188. rect.x = plane->x;
  189. rect.y = plane->y;
  190. /* Ask driver to update position only */
  191. rect.width = 0;
  192. rect.height = 0;
  193. err = plane->ops->update(plane, &rect);
  194. }
  195. }
  196. else
  197. {
  198. err = -RT_ENOSYS;
  199. }
  200. }
  201. else
  202. {
  203. err = -RT_EINVAL;
  204. }
  205. break;
  206. case RT_DEVICE_CTRL_CURSOR_SET_TYPE:
  207. if (args)
  208. {
  209. struct rt_graphic_plane *plane = gdev->cursor_plane;
  210. if (plane)
  211. {
  212. struct rt_device_rect_info rect =
  213. {
  214. .x = 0,
  215. .y = 0,
  216. .width = plane->width,
  217. .height = plane->height,
  218. };
  219. if (!plane->framebuffer)
  220. {
  221. rt_uint32_t mode;
  222. if (plane->mode == RTGRAPHIC_PIXEL_FORMAT_MONO)
  223. {
  224. mode = plane->modes[0];
  225. }
  226. else
  227. {
  228. mode = plane->mode;
  229. }
  230. if ((err = plane_fb_remap(plane, mode, &rect)))
  231. {
  232. break;
  233. }
  234. }
  235. rt_memcpy(plane->framebuffer, args, plane->screen_len);
  236. /* Force to update */
  237. err = plane->ops->update(plane, &rect);
  238. }
  239. else
  240. {
  241. err = -RT_ENOSYS;
  242. }
  243. }
  244. else
  245. {
  246. err = -RT_EINVAL;
  247. }
  248. break;
  249. case RTGRAPHIC_CTRL_RECT_UPDATE:
  250. if (args)
  251. {
  252. struct rt_graphic_plane *plane = plane_get_current(gdev);
  253. if (plane_need_update(plane))
  254. {
  255. err = plane->ops->update(plane, args);
  256. }
  257. }
  258. else
  259. {
  260. err = -RT_EINVAL;
  261. }
  262. break;
  263. case RTGRAPHIC_CTRL_POWERON:
  264. if (gdev->ops->dpms_switch)
  265. {
  266. err = graphic_dpms_switch(gdev, RT_GRAPHIC_DPMS_ON);
  267. }
  268. #ifdef RT_GRAPHIC_BACKLIGHT
  269. if (!err && gdev->backlight)
  270. {
  271. spin_unlock(&gdev->lock);
  272. err = rt_backlight_set_power(gdev->backlight, RT_BACKLIGHT_POWER_NORMAL);
  273. spin_lock(&gdev->lock);
  274. }
  275. #endif /* RT_GRAPHIC_BACKLIGHT */
  276. break;
  277. case RTGRAPHIC_CTRL_POWEROFF:
  278. if (gdev->ops->dpms_switch)
  279. {
  280. err = graphic_dpms_switch(gdev, RT_GRAPHIC_DPMS_OFF);
  281. }
  282. #ifdef RT_GRAPHIC_BACKLIGHT
  283. if (!err && gdev->backlight)
  284. {
  285. spin_unlock(&gdev->lock);
  286. err = rt_backlight_set_power(gdev->backlight, RT_BACKLIGHT_POWER_POWERDOWN);
  287. spin_lock(&gdev->lock);
  288. }
  289. #endif /* RT_GRAPHIC_BACKLIGHT */
  290. break;
  291. case RTGRAPHIC_CTRL_GET_INFO:
  292. if (args)
  293. {
  294. struct rt_device_graphic_info *info = args;
  295. struct rt_graphic_plane *plane = plane_get_current(gdev);
  296. info->pixel_format = plane->mode;
  297. info->bits_per_pixel = plane->bits_per_pixel;
  298. info->pitch = plane->line_length;
  299. info->width = plane->width;
  300. info->height = plane->height;
  301. info->framebuffer = plane->framebuffer;
  302. info->smem_len = plane->framebuffer_len;
  303. }
  304. else
  305. {
  306. err = -RT_EINVAL;
  307. }
  308. break;
  309. case RTGRAPHIC_CTRL_SET_MODE:
  310. if (args)
  311. {
  312. rt_uint32_t mode = (rt_uint32_t)(rt_ubase_t)args;
  313. struct rt_graphic_plane *plane = plane_get_current(gdev);
  314. if (mode != plane->mode)
  315. {
  316. err = -RT_ENOSYS;
  317. if (plane->modes_nr > 1)
  318. {
  319. for (int i = 0; i < plane->modes_nr; ++i)
  320. {
  321. if (mode == plane->modes[i])
  322. {
  323. struct rt_device_rect_info rect =
  324. {
  325. .x = plane->x,
  326. .y = plane->y,
  327. .width = plane->width,
  328. .height = plane->height,
  329. };
  330. err = plane_fb_remap(plane, mode, &rect);
  331. break;
  332. }
  333. }
  334. }
  335. }
  336. }
  337. else
  338. {
  339. err = -RT_EINVAL;
  340. }
  341. break;
  342. case RTGRAPHIC_CTRL_GET_EXT:
  343. if (args)
  344. {
  345. rt_memcpy(args, &gdev->edid, sizeof(gdev->edid));
  346. }
  347. else
  348. {
  349. err = -RT_EINVAL;
  350. }
  351. break;
  352. case RTGRAPHIC_CTRL_SET_BRIGHTNESS:
  353. if (gdev->ops->set_brightness)
  354. {
  355. err = gdev->ops->set_brightness(gdev, (rt_uint32_t)(rt_ubase_t)args);
  356. }
  357. #ifdef RT_GRAPHIC_BACKLIGHT
  358. if (!err && gdev->backlight)
  359. {
  360. spin_unlock(&gdev->lock);
  361. return rt_backlight_set_brightness(gdev->backlight, (rt_uint32_t)(rt_ubase_t)args);
  362. }
  363. #endif /* RT_GRAPHIC_BACKLIGHT */
  364. break;
  365. case RTGRAPHIC_CTRL_GET_BRIGHTNESS:
  366. if (args)
  367. {
  368. if (gdev->ops->get_brightness)
  369. {
  370. err = gdev->ops->get_brightness(gdev, args);
  371. }
  372. else
  373. {
  374. *(rt_uint32_t *)args = RT_UINT32_MAX >> 1;
  375. }
  376. }
  377. else
  378. {
  379. err = -RT_EINVAL;
  380. }
  381. #ifdef RT_GRAPHIC_BACKLIGHT
  382. if (!err && gdev->backlight)
  383. {
  384. spin_unlock(&gdev->lock);
  385. return rt_backlight_get_brightness(gdev->backlight, args);
  386. }
  387. #endif /* RT_GRAPHIC_BACKLIGHT */
  388. break;
  389. case RTGRAPHIC_CTRL_GET_MODE:
  390. if (args)
  391. {
  392. struct rt_graphic_plane *plane = plane_get_current(gdev);
  393. *(rt_uint32_t *)args = plane->mode;
  394. }
  395. else
  396. {
  397. err = -RT_EINVAL;
  398. }
  399. break;
  400. case RTGRAPHIC_CTRL_GET_STATUS:
  401. if (args)
  402. {
  403. if (gdev->ops->get_status)
  404. {
  405. err = gdev->ops->get_status(gdev, args);
  406. }
  407. else
  408. {
  409. err = -RT_ENOSYS;
  410. }
  411. }
  412. else
  413. {
  414. err = -RT_EINVAL;
  415. }
  416. break;
  417. case RTGRAPHIC_CTRL_PAN_DISPLAY:
  418. if (args)
  419. {
  420. struct rt_graphic_plane *plane = plane_get_current(gdev);
  421. if (plane->ops->fb_pan_display)
  422. {
  423. rt_size_t offset;
  424. struct rt_device_rect_info rect;
  425. offset = (rt_size_t)(args - plane->framebuffer);
  426. rect.x = (offset % plane->line_length) / (plane->bits_per_pixel / 8);
  427. rect.y = offset / plane->line_length;
  428. rect.width = plane->width;
  429. rect.height = plane->height;
  430. err = plane_fb_pan_display(plane, &rect);
  431. }
  432. else
  433. {
  434. err = -RT_ENOSYS;
  435. }
  436. }
  437. else
  438. {
  439. err = -RT_EINVAL;
  440. }
  441. break;
  442. case RTGRAPHIC_CTRL_WAIT_VSYNC:
  443. if (gdev->ops->wait_vsync)
  444. {
  445. err = gdev->ops->wait_vsync(gdev);
  446. }
  447. break;
  448. case RT_DEVICE_CTRL_NOTIFY_SET:
  449. if (args)
  450. {
  451. if (rt_atomic_load(&gdev->event_notifying) == RT_TRUE)
  452. {
  453. need_schedule = RT_TRUE;
  454. spin_unlock(&gdev->lock);
  455. goto _retry;
  456. }
  457. rt_memcpy(&gdev->event_notify, args, sizeof(gdev->event_notify));
  458. }
  459. else
  460. {
  461. err = -RT_EINVAL;
  462. }
  463. break;
  464. case FBIOGET_VSCREENINFO:
  465. if (args)
  466. {
  467. struct fb_var_screeninfo *var = args;
  468. struct rt_graphic_plane *plane = plane_get_current(gdev);
  469. rt_memset(var, 0, sizeof(*var));
  470. var->xres = plane->width;
  471. var->yres = plane->height;
  472. var->xres_virtual = plane->width;
  473. var->yres_virtual = plane->height * (plane->framebuffer_len / plane->screen_len);
  474. var->bits_per_pixel = plane->bits_per_pixel;
  475. if (plane == gdev->primary_plane)
  476. {
  477. var->width = gdev->edid.width_cm ? gdev->edid.width_cm * 10 : -1;
  478. var->height = gdev->edid.height_cm ? gdev->edid.height_cm * 10 : -1;
  479. }
  480. else
  481. {
  482. var->width = -1;
  483. var->height = -1;
  484. }
  485. if (plane->mode == RTGRAPHIC_PIXEL_FORMAT_GRAY4 ||
  486. plane->mode == RTGRAPHIC_PIXEL_FORMAT_GRAY16)
  487. {
  488. var->grayscale = 1;
  489. }
  490. else
  491. {
  492. const struct fb_format *fmt = &graphic_formats[0];
  493. for (int i = 0; i < RT_ARRAY_SIZE(graphic_formats); ++i, ++fmt)
  494. {
  495. if (fmt->mode == plane->mode)
  496. {
  497. rt_memcpy(&var->red, &fmt->red, sizeof(fmt->red));
  498. rt_memcpy(&var->green, &fmt->green, sizeof(fmt->green));
  499. rt_memcpy(&var->blue, &fmt->blue, sizeof(fmt->blue));
  500. rt_memcpy(&var->transp, &fmt->transp, sizeof(fmt->transp));
  501. break;
  502. }
  503. }
  504. }
  505. if (gdev->update_timer)
  506. {
  507. rt_uint64_t update_ps;
  508. rt_tick_t update_tick;
  509. rt_timer_control(gdev->update_timer, RT_TIMER_CTRL_GET_TIME, &update_tick);
  510. /*
  511. * 1s update_ms * 1000
  512. * -------------------- = ----------------- (second/tick)
  513. * RT_TICK_PER_SECOND update_tick
  514. *
  515. * 1000000ps = 1ms
  516. */
  517. update_ps = (update_tick * 1000000) / (RT_TICK_PER_SECOND * 1000);
  518. var->pixclock = update_ps / (var->xres * var->yres);
  519. }
  520. else
  521. {
  522. var->pixclock = (RT_GRAPHIC_UPDATE_MS * 1000000) / (var->xres * var->yres);
  523. }
  524. }
  525. else
  526. {
  527. err = -RT_EINVAL;
  528. }
  529. break;
  530. case FBIOPUT_VSCREENINFO:
  531. if (args)
  532. {
  533. rt_uint32_t mode;
  534. struct fb_format fmt;
  535. struct rt_device_rect_info rect;
  536. struct fb_var_screeninfo *var = args;
  537. struct rt_graphic_plane *plane = plane_get_current(gdev);
  538. const rt_size_t cmp_offset = rt_offsetof(struct fb_format, bits_per_pixel);
  539. const rt_size_t cmp_size = sizeof(struct fb_format) - cmp_offset;
  540. if (!plane->ops->fb_pan_display)
  541. {
  542. if (var->xres != plane->width || var->yres != plane->height ||
  543. var->xoffset != plane->x || var->yoffset != plane->y)
  544. {
  545. err = -RT_ENOSYS;
  546. break;
  547. }
  548. }
  549. mode = plane->mode;
  550. rect.x = plane->x;
  551. rect.y = plane->y;
  552. rect.width = plane->width;
  553. rect.height = plane->height;
  554. fmt.bits_per_pixel = var->bits_per_pixel;
  555. rt_memcpy(&fmt.red, &var->red, sizeof(fmt.red));
  556. rt_memcpy(&fmt.green, &var->green, sizeof(fmt.green));
  557. rt_memcpy(&fmt.blue, &var->blue, sizeof(fmt.blue));
  558. rt_memcpy(&fmt.transp, &var->transp, sizeof(fmt.transp));
  559. for (int i = 0; i < RT_ARRAY_SIZE(graphic_formats); ++i)
  560. {
  561. void *cmp_to = ((void *)&fmt) + cmp_offset;
  562. void *cmp_from = ((void *)&graphic_formats[i]) + cmp_offset;
  563. if (!rt_memcmp(cmp_to, cmp_from, cmp_size))
  564. {
  565. mode = graphic_formats[i].mode;
  566. break;
  567. }
  568. }
  569. err = -RT_ENOSYS;
  570. for (int i = 0; i < plane->modes_nr; ++i)
  571. {
  572. /* Check supported and commit */
  573. if (plane->modes[i] == mode && plane->mode != mode)
  574. {
  575. err = plane_fb_remap(plane, mode, &rect);
  576. }
  577. }
  578. if (!err && plane->ops->fb_pan_display)
  579. {
  580. rect.x = var->xoffset;
  581. rect.y = var->yoffset;
  582. rect.width = var->xres;
  583. rect.height = var->yres;
  584. err = plane_fb_pan_display(plane, &rect);
  585. }
  586. if (!err && var->rotate && plane->ops->prop_set)
  587. {
  588. err = plane->ops->prop_set(plane, RT_GRAPHIC_PLANE_PROP_ROTATE,
  589. (void *)(rt_ubase_t)((var->rotate % 360) / 90));
  590. }
  591. if (!err && plane == gdev->primary_plane && plane->ops->update)
  592. {
  593. rt_uint32_t update_ms = 0;
  594. if (var->pixclock)
  595. {
  596. rt_uint64_t clock_cycles;
  597. clock_cycles = var->pixclock;
  598. clock_cycles *= rect.width;
  599. clock_cycles *= rect.height;
  600. /* Seconds in pico seconds */
  601. clock_cycles /= 1000000000000ULL;
  602. update_ms = (rt_uint32_t)clock_cycles * 1000;
  603. }
  604. err = rt_graphic_device_update_auto(gdev, update_ms);
  605. }
  606. }
  607. else
  608. {
  609. err = -RT_EINVAL;
  610. }
  611. break;
  612. case FBIOGET_FSCREENINFO:
  613. if (args)
  614. {
  615. struct fb_fix_screeninfo *fix = args;
  616. struct rt_graphic_plane *plane = plane_get_current(gdev);
  617. rt_memset(fix, 0, sizeof(*fix));
  618. rt_snprintf(fix->id, rt_min_t(int, sizeof(fix->id), RT_NAME_MAX),
  619. "%s", rt_dm_dev_get_name(&gdev->parent));
  620. fix->smem_start = (unsigned long)rt_kmem_v2p(plane->framebuffer);
  621. fix->smem_len = plane->framebuffer_len;
  622. fix->mmio_start = fix->smem_start;
  623. fix->mmio_len = plane->screen_len;
  624. fix->line_length = plane->line_length;
  625. }
  626. else
  627. {
  628. err = -RT_EINVAL;
  629. }
  630. break;
  631. case FBIOGET_PIXELINFO:
  632. if (args)
  633. {
  634. struct rt_graphic_plane *plane = plane_get_current(gdev);
  635. *(rt_uint32_t *)args = plane->mode;
  636. }
  637. else
  638. {
  639. err = -RT_EINVAL;
  640. }
  641. break;
  642. case FBIOPAN_DISPLAY:
  643. if (args)
  644. {
  645. struct fb_var_screeninfo *var = args;
  646. struct rt_graphic_plane *plane = plane_get_current(gdev);
  647. if (plane->ops->fb_pan_display)
  648. {
  649. struct rt_device_rect_info rect;
  650. rect.x = var->xoffset;
  651. rect.y = var->yoffset;
  652. rect.width = var->xres;
  653. rect.height = var->yres;
  654. err = plane_fb_pan_display(plane, &rect);
  655. }
  656. else
  657. {
  658. err = -RT_ENOSYS;
  659. }
  660. }
  661. else
  662. {
  663. err = -RT_EINVAL;
  664. }
  665. break;
  666. case FBIO_CURSOR:
  667. err = -RT_EINVAL;
  668. break;
  669. case FBIOGET_CON2FBMAP:
  670. if (args)
  671. {
  672. struct fb_con2fbmap *con2fbmap = args;
  673. if (con2fbmap->console < RT_ARRAY_SIZE(fbcon_map))
  674. {
  675. con2fbmap->framebuffer = fbcon_map[con2fbmap->console];
  676. }
  677. else
  678. {
  679. err = -RT_EFULL;
  680. }
  681. }
  682. else
  683. {
  684. err = -RT_EINVAL;
  685. }
  686. break;
  687. case FBIOPUT_CON2FBMAP:
  688. if (args)
  689. {
  690. struct fb_con2fbmap *con2fbmap = args;
  691. if (con2fbmap->console < RT_ARRAY_SIZE(fbcon_map) &&
  692. con2fbmap->framebuffer < RT_ARRAY_SIZE(fbcon_map))
  693. {
  694. struct rt_device *vt;
  695. vt = rt_dm_device_find(MASTER_ID_TTY, con2fbmap->console);
  696. #ifdef RT_SERIAL_VIRTUAL
  697. if (!vt)
  698. {
  699. vt = rt_device_find("vuart");
  700. }
  701. #endif
  702. if (vt)
  703. {
  704. if (!(err = rt_device_open(vt, RT_DEVICE_OFLAG_RDWR)))
  705. {
  706. err = rt_device_control(vt, FBIOPUT_CON2FBMAP, con2fbmap);
  707. if (!err)
  708. {
  709. fbcon_map[con2fbmap->console] = con2fbmap->framebuffer;
  710. }
  711. rt_device_close(vt);
  712. }
  713. }
  714. else
  715. {
  716. err = -RT_EEMPTY;
  717. }
  718. }
  719. else
  720. {
  721. err = -RT_EFULL;
  722. }
  723. }
  724. else
  725. {
  726. err = -RT_EINVAL;
  727. }
  728. break;
  729. case FBIOBLANK:
  730. if (gdev->ops->dpms_switch)
  731. {
  732. rt_uint32_t dpms;
  733. switch ((rt_uint32_t)(rt_ubase_t)args)
  734. {
  735. case FB_BLANK_UNBLANK:
  736. /* Display: On, HSync: On, VSync: On */
  737. dpms = RT_GRAPHIC_DPMS_ON;
  738. break;
  739. case FB_BLANK_NORMAL:
  740. /* Display: Off, HSync: On, VSync: On */
  741. dpms = RT_GRAPHIC_DPMS_STANDBY;
  742. break;
  743. case FB_BLANK_HSYNC_SUSPEND:
  744. /* Display: Off, HSync: Off, VSync: On */
  745. dpms = RT_GRAPHIC_DPMS_STANDBY;
  746. break;
  747. case FB_BLANK_VSYNC_SUSPEND:
  748. /* Display: Off, HSync: On, VSync: Off */
  749. dpms = RT_GRAPHIC_DPMS_SUSPEND;
  750. break;
  751. case FB_BLANK_POWERDOWN:
  752. /* Display: Off, HSync: Off, VSync: Off */
  753. dpms = RT_GRAPHIC_DPMS_OFF;
  754. break;
  755. default:
  756. err = -RT_EINVAL;
  757. break;
  758. }
  759. if (!err)
  760. {
  761. graphic_dpms_switch(gdev, dpms);
  762. }
  763. }
  764. break;
  765. case FBIO_WAITFORVSYNC:
  766. if (gdev->ops->wait_vsync)
  767. {
  768. err = gdev->ops->wait_vsync(gdev);
  769. }
  770. break;
  771. case FBIO_ALLOC:
  772. case FBIO_FREE:
  773. case FBIOGET_GLYPH:
  774. case FBIOGET_HWCINFO:
  775. case FBIOPUT_MODEINFO:
  776. case FBIOGET_DISPINFO:
  777. LOG_D("FB IOCTL (%x) only used for SiS 300/630/730/540/315/550/650/740 frame buffer device", cmd);
  778. /* Fall through */
  779. case FBIOGET_VBLANK:
  780. case FBIOGETCMAP: /* fb_cmap */
  781. case FBIOPUTCMAP: /* fb_cmap */
  782. /* Fall through */
  783. default:
  784. if (gdev->ops->control)
  785. {
  786. err = gdev->ops->control(gdev, cmd, args);
  787. }
  788. else
  789. {
  790. err = -RT_ENOSYS;
  791. }
  792. }
  793. spin_unlock(&gdev->lock);
  794. return err;
  795. }
  796. #ifdef RT_USING_DEVICE_OPS
  797. const static struct rt_device_ops _graphic_ops =
  798. {
  799. .open = _graphic_open,
  800. .read = _graphic_read,
  801. .write = _graphic_write,
  802. .control = _graphic_control,
  803. };
  804. #endif
  805. static void graphic_ofw_init(struct rt_graphic_device *gdev)
  806. {
  807. #ifdef RT_USING_OFW
  808. struct rt_ofw_node *np = gdev->parent.ofw_node;
  809. #ifdef RT_GRAPHIC_BACKLIGHT
  810. if (!gdev->backlight)
  811. {
  812. struct rt_ofw_node *bl_np = rt_ofw_parse_phandle(np, "backlight", 0);
  813. if (bl_np && (gdev->backlight = rt_ofw_data(bl_np)))
  814. {
  815. rt_device_open(&gdev->backlight->parent, RT_DEVICE_OFLAG_RDWR);
  816. }
  817. rt_ofw_node_put(bl_np);
  818. }
  819. #endif /* RT_GRAPHIC_BACKLIGHT */
  820. (void)np;
  821. #endif /* RT_USING_OFW */
  822. }
  823. static void graphic_edid_res(struct rt_graphic_device *gdev,
  824. rt_uint32_t *out_width, rt_uint32_t *out_height)
  825. {
  826. struct edid *edid = &gdev->edid;
  827. struct detailed_timing *dt = &edid->detailed_timings[0];
  828. struct detailed_pixel_timing *dpt = &dt->data.pixel_data;
  829. *out_width = dpt->hactive_lo + ((dpt->hactive_hblank_hi & 0xf0) << 4);
  830. *out_height = dpt->vactive_lo + ((dpt->vactive_vblank_hi & 0xf0) << 4);
  831. }
  832. #ifdef RT_USING_PM
  833. static rt_err_t _graphic_pm_dpms_switch(struct rt_graphic_device *gdev, rt_uint8_t mode)
  834. {
  835. rt_err_t err;
  836. spin_lock(&gdev->lock);
  837. switch (mode)
  838. {
  839. case PM_SLEEP_MODE_NONE:
  840. case PM_SLEEP_MODE_IDLE:
  841. case PM_SLEEP_MODE_LIGHT:
  842. case PM_SLEEP_MODE_DEEP:
  843. err = graphic_dpms_switch(gdev, RT_GRAPHIC_DPMS_ON);
  844. break;
  845. case PM_SLEEP_MODE_STANDBY:
  846. err = graphic_dpms_switch(gdev, RT_GRAPHIC_DPMS_STANDBY);
  847. break;
  848. case PM_SLEEP_MODE_SHUTDOWN:
  849. err = graphic_dpms_switch(gdev, RT_GRAPHIC_DPMS_OFF);
  850. break;
  851. default:
  852. err = -RT_EINVAL;
  853. break;
  854. }
  855. spin_unlock(&gdev->lock);
  856. return err;
  857. }
  858. static rt_err_t _graphic_pm_suspend(const struct rt_device *device, rt_uint8_t mode)
  859. {
  860. struct rt_graphic_device *gdev = raw_to_graphic(device);
  861. return _graphic_pm_dpms_switch(gdev, mode);
  862. }
  863. static void _graphic_pm_resume(const struct rt_device *device, rt_uint8_t mode)
  864. {
  865. struct rt_graphic_device *gdev = raw_to_graphic(device);
  866. _graphic_pm_dpms_switch(gdev, mode);
  867. }
  868. static const struct rt_device_pm_ops _graphic_pm_ops =
  869. {
  870. .suspend = _graphic_pm_suspend,
  871. .resume = _graphic_pm_resume,
  872. };
  873. #endif /* RT_USING_PM */
  874. rt_err_t rt_graphic_device_register(struct rt_graphic_device *gdev)
  875. {
  876. rt_err_t err;
  877. int device_id;
  878. const char *dev_name;
  879. if (!gdev || !gdev->ops)
  880. {
  881. return -RT_EINVAL;
  882. }
  883. if (!gdev->primary_plane)
  884. {
  885. LOG_E("%s: Not %s found", rt_dm_dev_get_name(&gdev->parent), "primary plane");
  886. return -RT_EINVAL;
  887. }
  888. if ((device_id = rt_dm_ida_alloc(&graphic_ida)) < 0)
  889. {
  890. return -RT_EFULL;
  891. }
  892. rt_dm_dev_set_name(&gdev->parent, "fb%u", device_id);
  893. dev_name = rt_dm_dev_get_name(&gdev->parent);
  894. rt_list_init(&gdev->overlay_nodes);
  895. rt_dm_ida_init(&gdev->plane_ida, CUSTOM);
  896. rt_spin_lock_init(&gdev->lock);
  897. graphic_ofw_init(gdev);
  898. if (!gdev->primary_plane->width || !gdev->primary_plane->height)
  899. {
  900. rt_uint32_t mode, width, height;
  901. struct rt_device_rect_info rect;
  902. graphic_edid_res(gdev, &width, &height);
  903. rect.x = 0;
  904. rect.y = 0;
  905. rect.width = width;
  906. rect.height = height;
  907. if (gdev->primary_plane->mode == RTGRAPHIC_PIXEL_FORMAT_MONO)
  908. {
  909. mode = gdev->primary_plane->modes[0];
  910. }
  911. else
  912. {
  913. mode = gdev->primary_plane->mode;
  914. }
  915. err = plane_fb_remap(gdev->primary_plane, mode, &rect);
  916. if (err)
  917. {
  918. LOG_E("%s: Set %s error = %s", rt_dm_dev_get_name(&gdev->parent),
  919. "primary plane", rt_strerror(err));
  920. goto _fail;
  921. }
  922. }
  923. gdev->parent.type = RT_Device_Class_Graphic;
  924. #ifdef RT_USING_DEVICE_OPS
  925. gdev->parent.ops = &_graphic_ops;
  926. #else
  927. gdev->parent.open = _graphic_open;
  928. gdev->parent.read = _graphic_read;
  929. gdev->parent.write = _graphic_write;
  930. gdev->parent.control = _graphic_control;
  931. #endif
  932. gdev->parent.master_id = graphic_ida.master_id;
  933. gdev->parent.device_id = device_id;
  934. if ((err = rt_device_register(&gdev->parent, dev_name, RT_DEVICE_FLAG_RDWR)))
  935. {
  936. goto _fail;
  937. }
  938. #ifdef RT_USING_PM
  939. rt_pm_device_register(&gdev->parent, &_graphic_pm_ops);
  940. #endif
  941. if ((err = rt_graphic_logo_render(gdev)))
  942. {
  943. LOG_D("Logo render error = %s", rt_strerror(err));
  944. }
  945. return RT_EOK;
  946. _fail:
  947. rt_dm_ida_free(&graphic_ida, device_id);
  948. return err;
  949. }
  950. rt_err_t rt_graphic_device_unregister(struct rt_graphic_device *gdev)
  951. {
  952. const char *dev_name;
  953. struct rt_graphic_plane *plane, *plane_next;
  954. if (!gdev)
  955. {
  956. return -RT_EINVAL;
  957. }
  958. dev_name = rt_dm_dev_get_name(&gdev->parent);
  959. if (gdev->parent.ref_count)
  960. {
  961. LOG_E("%s: there is %u user", dev_name, gdev->parent.ref_count);
  962. return -RT_EINVAL;
  963. }
  964. #ifdef RT_USING_PM
  965. rt_pm_device_unregister(&gdev->parent);
  966. #endif
  967. rt_graphic_device_update_auto(gdev, 0);
  968. if (gdev->ops->dpms_switch)
  969. {
  970. graphic_dpms_switch(gdev, RT_GRAPHIC_DPMS_OFF);
  971. }
  972. #ifdef RT_GRAPHIC_BACKLIGHT
  973. if (gdev->backlight)
  974. {
  975. rt_backlight_set_power(gdev->backlight, RT_BACKLIGHT_POWER_POWERDOWN);
  976. rt_device_close(&gdev->backlight->parent);
  977. }
  978. #endif
  979. rt_list_for_each_entry_safe(plane, plane_next, &gdev->overlay_nodes, list)
  980. {
  981. rt_graphic_device_del_plane(gdev, plane);
  982. rt_graphic_device_free_plane(plane);
  983. }
  984. rt_graphic_device_del_plane(gdev, gdev->primary_plane);
  985. rt_graphic_device_free_plane(gdev->primary_plane);
  986. if (gdev->cursor_plane)
  987. {
  988. rt_graphic_device_del_plane(gdev, gdev->cursor_plane);
  989. rt_graphic_device_free_plane(gdev->cursor_plane);
  990. }
  991. rt_dm_ida_free(&graphic_ida, gdev->parent.device_id);
  992. rt_device_unregister(&gdev->parent);
  993. return RT_EOK;
  994. }
  995. struct rt_graphic_plane *rt_graphic_device_alloc_plane(struct rt_graphic_device *gdev,
  996. rt_size_t priv_size, const struct rt_graphic_plane_ops *ops,
  997. const rt_uint32_t *modes, rt_uint32_t modes_nr, rt_uint8_t type)
  998. {
  999. struct rt_graphic_plane *plane = RT_NULL;
  1000. if (!gdev || !ops || !modes || !modes_nr || type > RT_GRAPHIC_PLANE_TYPE_CURSOR)
  1001. {
  1002. return RT_NULL;
  1003. }
  1004. plane = rt_calloc(1, sizeof(*plane) + priv_size);
  1005. if (plane)
  1006. {
  1007. rt_list_init(&plane->list);
  1008. plane->type = type;
  1009. plane->modes_nr = modes_nr;
  1010. plane->modes = modes;
  1011. plane->mode = RTGRAPHIC_PIXEL_FORMAT_MONO;
  1012. plane->graphic = gdev;
  1013. plane->ops = ops;
  1014. }
  1015. return plane;
  1016. }
  1017. void rt_graphic_device_free_plane(struct rt_graphic_plane *plane)
  1018. {
  1019. if (!plane)
  1020. {
  1021. return;
  1022. }
  1023. rt_free(plane);
  1024. }
  1025. rt_err_t rt_graphic_device_add_plane(struct rt_graphic_device *gdev,
  1026. struct rt_graphic_plane *plane)
  1027. {
  1028. rt_err_t err = RT_EOK;
  1029. if (!gdev || !plane)
  1030. {
  1031. return -RT_EINVAL;
  1032. }
  1033. if (!plane->ops)
  1034. {
  1035. LOG_E("%s: %s have no plane ops",
  1036. rt_dm_dev_get_name(&gdev->parent), plane->name);
  1037. return -RT_EINVAL;
  1038. }
  1039. plane->id = rt_dm_ida_alloc(&gdev->plane_ida);
  1040. if (plane->id == RT_DM_IDA_NUM)
  1041. {
  1042. LOG_E("%s: %s is out of plane max(%d)",
  1043. rt_dm_dev_get_name(&gdev->parent), plane->name, RT_DM_IDA_NUM - 1);
  1044. return -RT_EFULL;
  1045. }
  1046. if (plane->type == RT_GRAPHIC_PLANE_TYPE_PRIMARY)
  1047. {
  1048. if (gdev->primary_plane)
  1049. {
  1050. err = -RT_EINVAL;
  1051. goto _free_ida;
  1052. }
  1053. if (!plane->name[0])
  1054. {
  1055. rt_strncpy(plane->name, "primary", sizeof(plane->name));
  1056. }
  1057. gdev->primary_plane = plane;
  1058. }
  1059. else if (plane->type == RT_GRAPHIC_PLANE_TYPE_CURSOR)
  1060. {
  1061. if (gdev->cursor_plane)
  1062. {
  1063. err = -RT_EINVAL;
  1064. goto _free_ida;
  1065. }
  1066. if (!plane->name[0])
  1067. {
  1068. rt_strncpy(plane->name, "cursor", sizeof(plane->name));
  1069. }
  1070. gdev->cursor_plane = plane;
  1071. }
  1072. else if (plane->type == RT_GRAPHIC_PLANE_TYPE_OVERLAY)
  1073. {
  1074. if (!plane->name[0])
  1075. {
  1076. rt_snprintf(plane->name, sizeof(plane->name), "overlay-%u", plane->id);
  1077. }
  1078. spin_lock(&gdev->lock);
  1079. rt_list_insert_before(&gdev->overlay_nodes, &plane->list);
  1080. rt_graphic_device_leave(gdev);
  1081. }
  1082. else
  1083. {
  1084. LOG_E("What the fuck plane type(%u)", plane->type);
  1085. RT_ASSERT(0);
  1086. }
  1087. _free_ida:
  1088. if (err)
  1089. {
  1090. rt_dm_ida_free(&gdev->plane_ida, plane->id);
  1091. }
  1092. return err;
  1093. }
  1094. rt_err_t rt_graphic_device_del_plane(struct rt_graphic_device *gdev,
  1095. struct rt_graphic_plane *plane)
  1096. {
  1097. if (!gdev || !plane)
  1098. {
  1099. return -RT_EINVAL;
  1100. }
  1101. if (plane->ops->fb_cleanup)
  1102. {
  1103. /* Ignore error */
  1104. plane->ops->fb_cleanup(plane);
  1105. }
  1106. if (plane->type == RT_GRAPHIC_PLANE_TYPE_PRIMARY)
  1107. {
  1108. gdev->primary_plane = RT_NULL;
  1109. }
  1110. else if (plane->type == RT_GRAPHIC_PLANE_TYPE_CURSOR)
  1111. {
  1112. gdev->cursor_plane = RT_NULL;
  1113. }
  1114. else if (plane->type == RT_GRAPHIC_PLANE_TYPE_OVERLAY)
  1115. {
  1116. spin_lock(&gdev->lock);
  1117. rt_list_remove(&plane->list);
  1118. spin_unlock(&gdev->lock);
  1119. }
  1120. rt_dm_ida_free(&gdev->plane_ida, plane->id);
  1121. return RT_EOK;
  1122. }
  1123. void rt_graphic_device_hotplug_event(struct rt_graphic_device *gdev)
  1124. {
  1125. rt_err_t err;
  1126. rt_uint32_t width, height;
  1127. struct rt_device_rect_info rect;
  1128. RT_ASSERT(gdev != RT_NULL);
  1129. rt_graphic_device_enter(gdev);
  1130. graphic_edid_res(gdev, &width, &height);
  1131. rect.x = 0;
  1132. rect.y = 0;
  1133. rect.width = width;
  1134. rect.height = height;
  1135. err = plane_fb_remap(gdev->primary_plane, gdev->primary_plane->mode, &rect);
  1136. if (err)
  1137. {
  1138. /* What the fuck? */
  1139. LOG_E("%s: hotplug event process error = %s",
  1140. rt_dm_dev_get_name(&gdev->parent), rt_strerror(err));
  1141. goto _out_lock;
  1142. }
  1143. _out_lock:
  1144. rt_graphic_device_leave(gdev);
  1145. rt_atomic_store(&gdev->event_notifying, RT_TRUE);
  1146. if (gdev->event_notify.notify)
  1147. {
  1148. gdev->event_notify.notify(gdev->event_notify.dev);
  1149. }
  1150. rt_atomic_store(&gdev->event_notifying, RT_FALSE);
  1151. }
  1152. static void graphic_device_plane_update(struct rt_graphic_plane *plane,
  1153. struct rt_device_rect_info *rect)
  1154. {
  1155. if (plane->ops->update)
  1156. {
  1157. plane->ops->update(plane, rect);
  1158. }
  1159. }
  1160. static void graphic_device_update(void *param)
  1161. {
  1162. struct rt_device_rect_info rect;
  1163. struct rt_graphic_plane *plane;
  1164. struct rt_graphic_device *gdev = param;
  1165. rect.x = 0;
  1166. rect.y = 0;
  1167. spin_lock(&gdev->lock);
  1168. rect.width = gdev->primary_plane->width;
  1169. rect.height = gdev->primary_plane->height;
  1170. graphic_device_plane_update(gdev->primary_plane, &rect);
  1171. rt_list_for_each_entry(plane, &gdev->overlay_nodes, list)
  1172. {
  1173. rect.width = plane->width;
  1174. rect.height = plane->height;
  1175. graphic_device_plane_update(plane, &rect);
  1176. }
  1177. if ((plane = gdev->cursor_plane))
  1178. {
  1179. rect.x = plane->x;
  1180. rect.y = plane->y;
  1181. /* Ask driver to update position only */
  1182. rect.width = 0;
  1183. rect.height = 0;
  1184. graphic_device_plane_update(plane, &rect);
  1185. }
  1186. spin_unlock(&gdev->lock);
  1187. }
  1188. rt_err_t rt_graphic_device_update_auto(struct rt_graphic_device *gdev, rt_uint32_t update_ms)
  1189. {
  1190. if (!gdev)
  1191. {
  1192. return -RT_EINVAL;
  1193. }
  1194. if (update_ms)
  1195. {
  1196. if (!gdev->update_timer)
  1197. {
  1198. char name[RT_NAME_MAX];
  1199. rt_snprintf(name, sizeof(name), "update-%s", rt_dm_dev_get_name(&gdev->parent));
  1200. gdev->update_timer = rt_timer_create(name, &graphic_device_update, gdev,
  1201. rt_tick_from_millisecond(update_ms),
  1202. RT_TIMER_FLAG_PERIODIC);
  1203. if (!gdev->update_timer)
  1204. {
  1205. return -RT_ENOMEM;
  1206. }
  1207. }
  1208. rt_timer_start(gdev->update_timer);
  1209. }
  1210. else if (gdev->update_timer)
  1211. {
  1212. rt_timer_stop(gdev->update_timer);
  1213. rt_timer_delete(gdev->update_timer);
  1214. gdev->update_timer = RT_NULL;
  1215. }
  1216. return RT_EOK;
  1217. }
  1218. void rt_graphic_device_enter(struct rt_graphic_device *gdev)
  1219. {
  1220. RT_ASSERT(gdev != RT_NULL);
  1221. spin_lock(&gdev->lock);
  1222. if (gdev->update_timer)
  1223. {
  1224. rt_timer_stop(gdev->update_timer);
  1225. }
  1226. }
  1227. void rt_graphic_device_leave(struct rt_graphic_device *gdev)
  1228. {
  1229. RT_ASSERT(gdev != RT_NULL);
  1230. if (gdev->update_timer)
  1231. {
  1232. rt_timer_start(gdev->update_timer);
  1233. }
  1234. spin_unlock(&gdev->lock);
  1235. }
  1236. rt_uint32_t rt_graphic_mode_bpp(rt_uint32_t mode)
  1237. {
  1238. for (int i = 0; i < RT_ARRAY_SIZE(graphic_formats); ++i)
  1239. {
  1240. if (graphic_formats[i].mode == mode)
  1241. {
  1242. return graphic_formats[i].bits_per_pixel;
  1243. }
  1244. }
  1245. return 0;
  1246. }