ark1668_lcdc_funcs.c 41 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338
  1. /*
  2. * Arkmicro ark1668 lcd driver
  3. *
  4. * Licensed under GPLv2 or later.
  5. */
  6. #include <linux/kernel.h>
  7. #include <linux/fb.h>
  8. #include <linux/poll.h>
  9. #include "ark1668_lcdc.h"
  10. #define DEBUG
  11. static int lcdc_width;
  12. static int lcdc_height;
  13. static void *lcdc_base = NULL;
  14. static struct ark1668_lcdfb_info *lcdfb_info;
  15. int ark1668_lcdc_funcs_init(struct ark1668_lcdfb_info *sinfo)
  16. {
  17. struct fb_info *info = sinfo->info;
  18. struct fb_var_screeninfo *var = &info->var;
  19. lcdfb_info = sinfo;
  20. lcdc_width = var->xres;
  21. lcdc_height = var->yres;
  22. lcdc_base = sinfo->mmio;
  23. return 0;
  24. }
  25. static void ark1668_lcdc_set_video_source_size(int id, int width, int height)
  26. {
  27. unsigned int val;
  28. int source_size_reg;
  29. if (id < 0 || id > VIDEO_LAYER2) return;
  30. val = ((height & 0xfff) << 12) | (width & 0xfff);
  31. switch (id) {
  32. case VIDEO_LAYER1:
  33. source_size_reg = ARK1668_LCDC_VIDEO_SOURCE_SIZE;
  34. break;
  35. case VIDEO_LAYER2:
  36. source_size_reg = ARK1668_LCDC_VIDEO2_SOURCE_SIZE;
  37. break;
  38. }
  39. writel(val, lcdc_base + source_size_reg);
  40. }
  41. static void ark1668_lcdc_set_video_layer_size(int id, int width, int height)
  42. {
  43. unsigned int val;
  44. int size_reg;
  45. if (id < 0 || id > VIDEO_LAYER2) return;
  46. val = ((height & 0xfff) << 12) | (width & 0xfff);
  47. switch (id) {
  48. case VIDEO_LAYER1:
  49. size_reg = ARK1668_LCDC_VIDEO_SIZE;
  50. break;
  51. case VIDEO_LAYER2:
  52. size_reg = ARK1668_LCDC_VIDEO2_SIZE;
  53. break;
  54. }
  55. writel(val, lcdc_base + size_reg);
  56. }
  57. static void ark1668_lcdc_set_video_layer_pos(int id, int x, int y)
  58. {
  59. int reg;
  60. int sign_x = 0, sign_y = 0;
  61. unsigned int val;
  62. if (id < 0 || id > VIDEO_LAYER2) return;
  63. if (x < 0) {
  64. sign_x = 1;
  65. x = -x;
  66. }
  67. if (y < 0) {
  68. sign_y = 1;
  69. y = -y;
  70. }
  71. x &= 0xfff;
  72. y &= 0xfff;
  73. switch (id) {
  74. case VIDEO_LAYER1:
  75. reg = ARK1668_LCDC_VIDEO_POSITION;
  76. break;
  77. case VIDEO_LAYER2:
  78. reg = ARK1668_LCDC_VIDEO2_POSITION;
  79. break;
  80. }
  81. val = (sign_y << 25) | (y << 13) | (sign_x << 12) | x;
  82. writel(val, lcdc_base + reg);
  83. }
  84. static void ark1668_lcdc_set_video_win_size(int id, int width, int height)
  85. {
  86. unsigned int val;
  87. int size_reg;
  88. if (id < 0 || id > VIDEO_LAYER2) return;
  89. val = ((height & 0xfff) << 12) | (width & 0xfff);
  90. switch (id) {
  91. case VIDEO_LAYER1:
  92. size_reg = ARK1668_LCDC_VIDEO_WIN_SIZE;
  93. break;
  94. case VIDEO_LAYER2:
  95. size_reg = ARK1668_LCDC_VIDEO2_WIN_SIZE;
  96. break;
  97. }
  98. writel(val, lcdc_base + size_reg);
  99. }
  100. static void ark1668_lcdc_set_video_win_point(int id, int x, int y)
  101. {
  102. unsigned int val;
  103. int reg;
  104. if (id < 0 || id > VIDEO_LAYER2) return;
  105. val = ((y & 0xfff) << 12) | (x & 0xfff);
  106. switch (id) {
  107. case VIDEO_LAYER1:
  108. reg = ARK1668_LCDC_VIDEO_WIN_POINT;
  109. break;
  110. case VIDEO_LAYER2:
  111. reg = ARK1668_LCDC_VIDEO2_WIN_POINT;
  112. break;
  113. }
  114. writel(val, lcdc_base + reg);
  115. }
  116. static void ark1668_lcdc_set_video_en(int id, int enable)
  117. {
  118. unsigned int val;
  119. int offset;
  120. if (id < 0 || id > VIDEO_LAYER2) return;
  121. val = readl(lcdc_base + ARK1668_LCDC_CONTROL);
  122. switch (id) {
  123. case VIDEO_LAYER1:
  124. offset = 5;
  125. break;
  126. case VIDEO_LAYER2:
  127. offset = 6;
  128. break;
  129. }
  130. if (enable)
  131. val |= (1 << offset);
  132. else
  133. val &= ~(1 << offset);
  134. writel(val, lcdc_base + ARK1668_LCDC_CONTROL);
  135. }
  136. static void ark1668_lcdc_set_video_format(int id, unsigned int format,
  137. unsigned int yuyv_order, unsigned int rgb_order, unsigned int scal_bypass)
  138. {
  139. unsigned int yuv_ycbcr_bypass, rgb_ycbcr_bypass;
  140. unsigned int y_uv_order;
  141. unsigned int pixfmt;
  142. int reg;
  143. unsigned val;
  144. if (id < 0 || id > VIDEO_LAYER2) return;
  145. switch (id) {
  146. case VIDEO_LAYER1:
  147. reg = ARK1668_LCDC_VIDEO_CTL;
  148. break;
  149. case VIDEO_LAYER2:
  150. reg = ARK1668_LCDC_VIDEO2_CTL;
  151. break;
  152. }
  153. switch (format)
  154. {
  155. case ARK_LCDC_FORMAT_RGBI555:
  156. case ARK_LCDC_FORMAT_R5G6B5:
  157. case ARK_LCDC_FORMAT_RGBA888:
  158. case ARK_LCDC_FORMAT_RGB888:
  159. yuv_ycbcr_bypass = 1;
  160. rgb_ycbcr_bypass = 0;
  161. y_uv_order = 0;
  162. break;
  163. case ARK_LCDC_FORMAT_Y_UV422:
  164. case ARK_LCDC_FORMAT_Y_UV420:
  165. yuv_ycbcr_bypass = 1;
  166. rgb_ycbcr_bypass = 1;
  167. y_uv_order = 1;
  168. break;
  169. case ARK_LCDC_FORMAT_VYUY:
  170. yuv_ycbcr_bypass = 0;
  171. rgb_ycbcr_bypass = 0;
  172. y_uv_order = 0;
  173. break;
  174. default:
  175. yuv_ycbcr_bypass = 1;
  176. rgb_ycbcr_bypass = 1;
  177. y_uv_order = 0;
  178. break;
  179. }
  180. pixfmt = format & 0x0F;
  181. //val = readl(lcdc_base + reg);
  182. if (id == 0) {
  183. val =
  184. 0<<23 | // 0=disable to get more data bytes on each line
  185. 1<<22 | // 1=select video size specified on CLCD_VIDEO_SIZE
  186. y_uv_order<<21 | // 1=Y_UV, 0=Y_U_V
  187. 0<<20 | // 0=double buffer mode disabled
  188. 0<<19 | // 0=use address group 0 for display
  189. yuyv_order<<17 | // 18-17: 0=yuyv in case YUV422
  190. rgb_order<<14 | // 16-14: 000=RGB data order or YUV order for YUV444
  191. 0<<11 | // 0=do not keep UV data for next line in case
  192. 0<<10 | // 0=use 2-step FIR low-pass filter to do data transfer
  193. scal_bypass<<9 | // 1=enable video scalar by pass mode
  194. 1<<8 | // 1=whole scaler is disable when scaler is in bypass mode
  195. 0<<7 | // 0=field mode for interlace data capture of video
  196. 0<<6 | // 0=not invert field
  197. yuv_ycbcr_bypass<<5 | // 1=no YUV to YCbCr, 0=YUV to YCbCr
  198. rgb_ycbcr_bypass<<4 | // 1=no RGB to YCbCr, 0=RGB to YCbCr
  199. pixfmt<<0; // 3-0: video layer source format
  200. if (pixfmt == ARK_LCDC_FORMAT_YUV420)
  201. val |= 1<<11;
  202. } else if (id == 1) {
  203. val =
  204. 0<<26 | // 0=u/v write first in doing data write back
  205. 0<<25 | // 0=v first in doing data write back
  206. 0<<24 | // wb_start, set this bit from 0->1 will start write back
  207. 0<<23 | // 0=disable to get more data bytes for each line read
  208. 1<<22 | // 1= select video size on REG0x330 rLCD_VIDEO2_SIZE
  209. y_uv_order<<21 | // 0=Y_U_V, 1=Y_UV
  210. 0<<20 | // 0=double buffer mode disable
  211. 0<<19 | // 0=use addr group 0 when double buffer mode disable
  212. yuyv_order<<17 | // 18-17: 00=yuyv when format is YUV422
  213. rgb_order<<14 | // 16-14: 000=RGB when format is YUV444 or RGB
  214. 0<<13 | // 0=not save UV data for displaying next line
  215. 0<<12 | // 0=use 2-step FIR low-pass filter for processing data
  216. 0<<11 | // 0=use axi_clk
  217. 0<<10 | // 0=?v????? progressive read in doing write back
  218. scal_bypass<<9 | // 1=enable scale by pass mode
  219. 1<<8 | // 1=disable whole scale in video layer when scaler is
  220. 0<<7 | // 0=field mode select for interlace data capture of
  221. 0<<6 | // 0=field not invert for frame mode and interlace data
  222. yuv_ycbcr_bypass<<5 | // 1=no yuv2ycbcr
  223. rgb_ycbcr_bypass<<4 | // 1=no rgb2ycbcr, 0=do rgb2ycbcr
  224. pixfmt<<0; // 3-0: video layer src format
  225. if (pixfmt == ARK_LCDC_FORMAT_YUV420)
  226. val |= 1<<13;
  227. }
  228. writel(val, lcdc_base + reg);
  229. }
  230. static void ark1668_lcdc_set_video_scal(
  231. int id,
  232. unsigned int win_width, unsigned int win_height,
  233. unsigned int left_blank, unsigned int right_blank,
  234. unsigned int top_blank, unsigned int bottom_blank,
  235. unsigned int dst_width, unsigned int dst_height,
  236. int interlace_out_en,
  237. int interlace_read_en // 1=interlace, 0=progressive
  238. )
  239. {
  240. unsigned int vblank = top_blank + bottom_blank;
  241. unsigned int hblank = left_blank + right_blank;
  242. unsigned int reg_ctl, reg_ctl0, reg_ctl1, reg_cut;
  243. unsigned int val;
  244. if (id < 0 || id > VIDEO_LAYER2) return;
  245. if (dst_width == 0 || dst_height == 0)
  246. return;
  247. switch (id) {
  248. case VIDEO_LAYER1:
  249. reg_ctl = ARK1668_LCDC_VIDEO_SCALE_CTL;
  250. reg_ctl0 = ARK1668_LCDC_VIDEO_SCAL_CTL0;
  251. reg_ctl1 = ARK1668_LCDC_VIDEO_SCAL_CTL1;
  252. reg_cut = ARK1668_LCDC_VIDEO_RIGHT_BOTTOM_CUT_NUM;
  253. break;
  254. case VIDEO_LAYER2:
  255. reg_ctl = ARK1668_LCDC_VIDEO2_SCALE_CTL;
  256. reg_ctl0 = ARK1668_LCDC_VIDEO2_SCAL_CTL0;
  257. reg_ctl1 = ARK1668_LCDC_VIDEO2_SCAL_CTL1;
  258. reg_cut = ARK1668_LCDC_VIDEO2_RIGHT_BOTTOM_CUT_NUM;
  259. break;
  260. }
  261. val = 0<<11| // 0=addr update per field
  262. 0<<9 | // 10-9: 00=
  263. 0<<8 | // 0=not line chroma
  264. 1<<7 | // 1=YUV
  265. 0<<6 | // 0=disable horizontal filter (use for down scale)
  266. 1<<5 | // 1=auto set coef of h-filter when down scale
  267. 0<<4 | // 0=normal scale de-interlace mode
  268. 0<<3 | // 0=current field is field=0
  269. 0<<2 | // 0=field=0 is odd, field=1 is even
  270. 0<<1 | // 0=de-interlace disable
  271. 0<<0; // 0=use 2 line buffers
  272. if ((dst_width + hblank) * 3 / 2 < win_width)
  273. val |= 1<<6;
  274. writel(val, lcdc_base + reg_ctl);
  275. val = right_blank<<8 | bottom_blank;
  276. writel(val, lcdc_base + reg_cut);
  277. val = left_blank<<18 |(win_width * 1024 / (dst_width + hblank));
  278. writel(val, lcdc_base + reg_ctl0);
  279. val = top_blank<<18 | (win_height * 1024 / (dst_height + vblank));
  280. writel(val, lcdc_base + reg_ctl1);
  281. if ((val & 0x3FFFF) == 0x400){
  282. val = readl(lcdc_base + reg_ctl);
  283. val |= (1 << 12) | (1 << 0);
  284. writel(val, lcdc_base + reg_ctl);
  285. }
  286. if (interlace_out_en) {
  287. val = readl(lcdc_base + ARK1668_LCDC_TV_CONTROL);
  288. val &= ~(1<<8);
  289. writel(val, lcdc_base + ARK1668_LCDC_TV_CONTROL);
  290. val = readl(lcdc_base + reg_ctl1);
  291. if ((val & 0x3FFFF) == 0x400){
  292. val -= 1;
  293. writel(val, lcdc_base + reg_ctl1);
  294. }
  295. val = readl(lcdc_base + reg_ctl);
  296. val &= ~(7<<9);
  297. val |= 1<<11 | 1<<9;
  298. writel(val, lcdc_base + reg_ctl);
  299. }
  300. }
  301. void ark1668_lcdc_set_video_addr(int id, unsigned int yaddr,unsigned int cbaddr, unsigned int craddr)
  302. {
  303. unsigned int reg_addr1, reg_addr2, reg_addr3;
  304. if (id < 0 || id > VIDEO_LAYER2) return;
  305. switch (id) {
  306. case VIDEO_LAYER1:
  307. reg_addr1 = ARK1668_LCDC_VIDEO_ADDR1;
  308. reg_addr2 = ARK1668_LCDC_VIDEO_ADDR2;
  309. reg_addr3 = ARK1668_LCDC_VIDEO_ADDR3;
  310. break;
  311. case VIDEO_LAYER2:
  312. reg_addr1 = ARK1668_LCDC_VIDEO2_ADDR1;
  313. reg_addr2 = ARK1668_LCDC_VIDEO2_ADDR2;
  314. reg_addr3 = ARK1668_LCDC_VIDEO2_ADDR3;
  315. break;
  316. }
  317. writel(yaddr, lcdc_base + reg_addr1);
  318. if(cbaddr) writel(cbaddr, lcdc_base + reg_addr2);
  319. if(craddr) writel(craddr, lcdc_base + reg_addr3);
  320. }
  321. static void ark1668_lcdc_get_video_addr(int id, unsigned int *yaddr,
  322. unsigned int *cbaddr, unsigned int *craddr)
  323. {
  324. if (id < 0 || id > VIDEO_LAYER2) return;
  325. switch (id) {
  326. case VIDEO_LAYER1:
  327. *yaddr = readl(lcdc_base + ARK1668_LCDC_VIDEO_ADDR1);
  328. *cbaddr = readl(lcdc_base + ARK1668_LCDC_VIDEO_ADDR2);
  329. *craddr = readl(lcdc_base + ARK1668_LCDC_VIDEO_ADDR3);
  330. break;
  331. case VIDEO_LAYER2:
  332. *yaddr = readl(lcdc_base + ARK1668_LCDC_VIDEO2_ADDR1);
  333. *cbaddr = readl(lcdc_base + ARK1668_LCDC_VIDEO2_ADDR2);
  334. *craddr = readl(lcdc_base + ARK1668_LCDC_VIDEO2_ADDR3);
  335. break;
  336. }
  337. }
  338. static void ark1668_lcdc_set_osd_size(int id, int width, int height)
  339. {
  340. unsigned int val;
  341. int size_reg, source_size_reg;
  342. if (id < 0 || id > OSD_LAYER3) return;
  343. val = ((height & 0xfff) << 12) | (width & 0xfff);
  344. switch (id) {
  345. case OSD_LAYER1:
  346. size_reg = ARK1668_LCDC_OSD1_SIZE;
  347. source_size_reg = ARK1668_LCDC_OSD1_SOURCE_SIZE;
  348. break;
  349. case OSD_LAYER2:
  350. size_reg = ARK1668_LCDC_OSD2_SIZE;
  351. source_size_reg = ARK1668_LCDC_OSD2_SOURCE_SIZE;
  352. break;
  353. case OSD_LAYER3:
  354. size_reg = ARK1668_LCDC_OSD3_SIZE;
  355. source_size_reg = ARK1668_LCDC_OSD3_SOURCE_SIZE;
  356. break;
  357. }
  358. writel(val, lcdc_base + size_reg);
  359. writel(val, lcdc_base + source_size_reg);
  360. }
  361. static void ark1668_lcdc_set_osd_pos(int id, int x, int y)
  362. {
  363. int reg;
  364. int sign_x = 0, sign_y = 0;
  365. unsigned int val;
  366. if (id < 0 || id > OSD_LAYER3) return;
  367. if (x < 0) {
  368. sign_x = 1;
  369. x = -x;
  370. }
  371. if (y < 0) {
  372. sign_y = 1;
  373. y = -y;
  374. }
  375. x &= 0xfff;
  376. y &= 0xfff;
  377. switch (id) {
  378. case OSD_LAYER1:
  379. reg = ARK1668_LCDC_OSD1_POS;
  380. break;
  381. case OSD_LAYER2:
  382. reg = ARK1668_LCDC_OSD2_POS;
  383. break;
  384. case OSD_LAYER3:
  385. reg = ARK1668_LCDC_OSD3_POS;
  386. break;
  387. }
  388. val = (sign_y << 25) | (y << 13) | (sign_x << 12) | x;
  389. writel(val, lcdc_base + reg);
  390. }
  391. static void ark1668_lcdc_set_osd_format(int id, int format, int yuv_order, int rgb_order)
  392. {
  393. int reg;
  394. unsigned val;
  395. int yuv_ycbcr_bypass, rgb_ycbcr_bypass;
  396. if (id < 0 || id > OSD_LAYER3) return;
  397. switch (format) {
  398. case ARK1668_LCDC_FORMAT_VYUY:
  399. yuv_ycbcr_bypass = 1;
  400. rgb_ycbcr_bypass = 1;
  401. break;
  402. case ARK1668_LCDC_FORMAT_YUV:
  403. yuv_ycbcr_bypass = 1;
  404. rgb_ycbcr_bypass = 1;
  405. break;
  406. case ARK1668_LCDC_FORMAT_RGBI555:
  407. case ARK1668_LCDC_FORMAT_R5G6B5:
  408. case ARK1668_LCDC_FORMAT_RGBA888:
  409. case ARK1668_LCDC_FORMAT_RGB888:
  410. yuv_ycbcr_bypass = 1;
  411. rgb_ycbcr_bypass = 0;
  412. break;
  413. default:
  414. return;
  415. }
  416. switch (id) {
  417. case OSD_LAYER1:
  418. reg = ARK1668_LCDC_OSD1_CTL;
  419. break;
  420. case OSD_LAYER2:
  421. reg = ARK1668_LCDC_OSD2_CTL;
  422. break;
  423. case OSD_LAYER3:
  424. reg = ARK1668_LCDC_OSD3_CTL;
  425. break;
  426. }
  427. val = readl(lcdc_base + reg);
  428. val &= ~(0x7ff << 12);
  429. val |= ((yuv_order & 3) << 21) | ((rgb_order & 7) << 18) | (yuv_ycbcr_bypass << 17) |
  430. (rgb_ycbcr_bypass << 16) | ((format & 0xf) << 12);
  431. writel(val, lcdc_base + reg);
  432. }
  433. void ark1668_lcdc_set_osd_addr(int id, unsigned int addr)
  434. {
  435. int reg;
  436. if (id < 0 || id > OSD_LAYER3) return;
  437. switch (id) {
  438. case OSD_LAYER1:
  439. reg = ARK1668_LCDC_OSD1_ADDR;
  440. break;
  441. case OSD_LAYER2:
  442. reg = ARK1668_LCDC_OSD2_ADDR;
  443. break;
  444. case OSD_LAYER3:
  445. reg = ARK1668_LCDC_OSD3_ADDR;
  446. break;
  447. }
  448. writel(addr, lcdc_base + reg);
  449. }
  450. unsigned int ark1668_lcdc_get_osd_addr(int id)
  451. {
  452. int reg;
  453. if (id < 0 || id > OSD_LAYER3) return 0;
  454. switch (id) {
  455. case OSD_LAYER1:
  456. reg = ARK1668_LCDC_OSD1_ADDR;
  457. break;
  458. case OSD_LAYER2:
  459. reg = ARK1668_LCDC_OSD2_ADDR;
  460. break;
  461. case OSD_LAYER3:
  462. reg = ARK1668_LCDC_OSD3_ADDR;
  463. break;
  464. }
  465. return readl(lcdc_base + reg);
  466. }
  467. static void ark1668_lcdc_set_osd_en(int id, int enable)
  468. {
  469. unsigned int val;
  470. int offset;
  471. if (id < 0 || id > OSD_LAYER3) return;
  472. val = readl(lcdc_base + ARK1668_LCDC_CONTROL);
  473. switch (id) {
  474. case OSD_LAYER1:
  475. offset = 7;
  476. break;
  477. case OSD_LAYER2:
  478. offset = 8;
  479. break;
  480. case OSD_LAYER3:
  481. offset = 9;
  482. break;
  483. }
  484. if (enable)
  485. val |= (1 << offset);
  486. else
  487. val &= ~(1 << offset);
  488. writel(val, lcdc_base + ARK1668_LCDC_CONTROL);
  489. }
  490. static void ark1668_lcdc_set_video_priority(int level)
  491. {
  492. unsigned int val;
  493. val = readl(lcdc_base + ARK1668_LCDC_MODE_LCD_REG0);
  494. val &= ~(0x7<<0);
  495. val |= (level<<0);
  496. writel(val, lcdc_base + ARK1668_LCDC_MODE_LCD_REG0);
  497. }
  498. static void ark1668_lcdc_set_video2_priority(int level)
  499. {
  500. unsigned int val;
  501. val = readl(lcdc_base + ARK1668_LCDC_MODE_LCD_REG1);
  502. val &= ~(0x7<<0);
  503. val |= (level<<0);
  504. writel(val, lcdc_base + ARK1668_LCDC_MODE_LCD_REG1);
  505. }
  506. static void ark1668_lcdc_set_win1_priority(int level)
  507. {
  508. unsigned int val;
  509. val = readl(lcdc_base + ARK1668_LCDC_MODE_LCD_REG0);
  510. val &= ~(0x7<<8);
  511. val |= (level<<8);
  512. writel(val, lcdc_base + ARK1668_LCDC_MODE_LCD_REG0);
  513. }
  514. static void ark1668_lcdc_set_win2_priority(int level)
  515. {
  516. unsigned int val;
  517. val = readl(lcdc_base + ARK1668_LCDC_MODE_LCD_REG0);
  518. val &= ~(0x7<<16);
  519. val |= (level<<16);
  520. writel(val, lcdc_base + ARK1668_LCDC_MODE_LCD_REG0);
  521. }
  522. static void ark1668_lcdc_set_win3_priority(int level)
  523. {
  524. unsigned int val;
  525. val = readl(lcdc_base + ARK1668_LCDC_MODE_LCD_REG0);
  526. val &= ~(0x7<<24);
  527. val |= (level<<24);
  528. writel(val, lcdc_base + ARK1668_LCDC_MODE_LCD_REG0);
  529. }
  530. static void ark1668_lcdc_set_osd_alpha_blend_en_lcd(int id, int enable)
  531. {
  532. unsigned int pos;
  533. unsigned int val;
  534. if (id < 0 || id > OSD_LAYER3) return;
  535. val = readl(lcdc_base + ARK1668_LCDC_MODE_LCD_REG1);
  536. switch (id)
  537. {
  538. case OSD_LAYER1:
  539. pos = 13;
  540. break;
  541. case OSD_LAYER2:
  542. pos = 15;
  543. break;
  544. case OSD_LAYER3:
  545. pos = 17;
  546. break;
  547. default:
  548. return;
  549. }
  550. if (enable)
  551. val |= 1 << pos;
  552. else
  553. val &= ~(1 << pos);
  554. writel(val, lcdc_base + ARK1668_LCDC_MODE_LCD_REG1);
  555. }
  556. static void ark1668_lcdc_set_osd_per_pix_alpha_blend_en_lcd(int id, int enable)
  557. {
  558. unsigned int pos;
  559. unsigned int val;
  560. if (id < 0 || id > OSD_LAYER3) return;
  561. val = readl(lcdc_base + ARK1668_LCDC_MODE_LCD_REG1);
  562. switch (id)
  563. {
  564. case OSD_LAYER1:
  565. pos = 12;
  566. break;
  567. case OSD_LAYER2:
  568. pos = 14;
  569. break;
  570. case OSD_LAYER3:
  571. pos = 16;
  572. break;
  573. default:
  574. return;
  575. }
  576. if (enable)
  577. val |= 1 << pos;
  578. else
  579. val &= ~(1 << pos);
  580. writel(val, lcdc_base + ARK1668_LCDC_MODE_LCD_REG1);
  581. }
  582. int ark_bootanimation_display_init(int width, int height, unsigned int addr)
  583. {
  584. ark1668_lcdc_set_osd_size(OSD_LAYER2, width, height);
  585. ark1668_lcdc_set_osd_pos(OSD_LAYER2, (lcdc_width - width) / 2, (lcdc_height - height) / 2);
  586. ark1668_lcdc_set_osd_format(OSD_LAYER2, ARK1668_LCDC_FORMAT_VYUY, ARK_LCDC_ORDER_UYVY, 0);
  587. ark1668_lcdc_set_osd_addr(OSD_LAYER2, addr);
  588. ark1668_lcdc_set_osd_en(OSD_LAYER2, 1);
  589. return 0;
  590. }
  591. EXPORT_SYMBOL(ark_bootanimation_display_init);
  592. int ark_bootanimation_display_uninit(void)
  593. {
  594. ark1668_lcdc_set_osd_en(OSD_LAYER2, 0);
  595. return 0;
  596. }
  597. EXPORT_SYMBOL(ark_bootanimation_display_uninit);
  598. int ark_bootanimation_set_display_addr(unsigned int addr)
  599. {
  600. ark1668_lcdc_set_osd_addr(OSD_LAYER2, addr);
  601. ark1668_lcdc_wait_for_vsync();
  602. return 0;
  603. }
  604. EXPORT_SYMBOL(ark_bootanimation_set_display_addr);
  605. int ark_itu656_display_init(int src_width, int src_height,int out_posx, int out_posy,int out_width, int out_height,int interlace)
  606. {
  607. int win_x, win_y, win_width, win_height;
  608. if(lcdc_base == NULL)
  609. return -1;
  610. if(interlace){
  611. win_x = 0;
  612. win_y = 2;
  613. win_width = src_width - 12;
  614. win_height = src_height - win_y -4;
  615. }else{
  616. win_x = 2;
  617. win_y = 2;
  618. win_width = src_width - win_x -2;
  619. win_height = src_height;
  620. }
  621. ark1668_lcdc_set_video_source_size(VIDEO_LAYER2, src_width, src_height);
  622. ark1668_lcdc_set_video_win_size(VIDEO_LAYER2, win_width, win_height);
  623. ark1668_lcdc_set_video_win_point(VIDEO_LAYER2, 0, 0);
  624. ark1668_lcdc_set_video_layer_pos(VIDEO_LAYER2, out_posx, out_posy);
  625. ark1668_lcdc_set_video_layer_size(VIDEO_LAYER2, out_width, out_height);
  626. ark1668_lcdc_set_video_format(VIDEO_LAYER2, ARK_LCDC_FORMAT_VYUY, 0 ,ARK_LCDC_ORDER_VYUY, 0);
  627. ark1668_lcdc_set_video_scal(VIDEO_LAYER2, win_width, win_height, 0, 0, 0, 0, out_width, out_height, 0, 0);
  628. return 0;
  629. }
  630. EXPORT_SYMBOL(ark_itu656_display_init);
  631. int ark_itu656_display_addr(unsigned int addr)
  632. {
  633. if(lcdc_base == NULL || addr == 0){
  634. return -1;
  635. }
  636. ark1668_lcdc_set_video_addr(VIDEO_LAYER2, addr, 0, 0);
  637. return 0;
  638. }
  639. EXPORT_SYMBOL(ark_itu656_display_addr);
  640. int ark_itu656_display_uninit(void)
  641. {
  642. if(lcdc_base == NULL){
  643. return -1;
  644. }
  645. ark1668_lcdc_set_video_en(VIDEO_LAYER2, 0);
  646. return 0;
  647. }
  648. EXPORT_SYMBOL(ark_itu656_display_uninit);
  649. int ark_disp_set_layer_en(int layer_id, int enable)
  650. {
  651. if(lcdc_base == NULL){
  652. return -1;
  653. }
  654. if(layer_id > 4 || layer_id < 0){
  655. return -1;
  656. }
  657. if(layer_id < 3){
  658. ark1668_lcdc_set_osd_en(layer_id, enable);
  659. }else{
  660. ark1668_lcdc_set_video_en(layer_id-3, enable);
  661. }
  662. return 0;
  663. }
  664. EXPORT_SYMBOL(ark_disp_set_layer_en);
  665. int ark168vin_set_scal(int layer,int src_w, int src_h,int out_w,int out_h)
  666. {
  667. ark1668_lcdc_set_video_scal(layer-OSD_LAYER_MAX, src_w, src_h, 0, 0, 0, 0, out_w, out_h, 0, 0);
  668. return 0;
  669. }
  670. EXPORT_SYMBOL(ark168vin_set_scal);
  671. int ark168vin_set_display(int layer, unsigned int cmd, void *arg)
  672. {
  673. unsigned long error = 0;
  674. switch (cmd) {
  675. case VIN_SHOW_WINDOW:
  676. {
  677. if(layer <= OSD_LAYER3)
  678. ark1668_lcdc_set_osd_en(layer, 1);
  679. else
  680. ark1668_lcdc_set_video_en(layer-OSD_LAYER_MAX, 1);
  681. printk(KERN_DEBUG "layer=%d: show window.\n ",layer);
  682. break;
  683. }
  684. case VIN_HIDE_WINDOW:
  685. {
  686. if(layer <= OSD_LAYER3)
  687. ark1668_lcdc_set_osd_en(layer, 0);
  688. else
  689. ark1668_lcdc_set_video_en(layer-OSD_LAYER_MAX, 0);
  690. printk(KERN_DEBUG "layer=%d: hide window.\n ",layer);
  691. break;
  692. }
  693. case VIN_SET_WINDOW_POS:
  694. {
  695. unsigned int x,y,data;
  696. data = *(unsigned int*)arg;
  697. x = data & 0xFFFF;
  698. y = (data >> 16) & 0xFFFF;
  699. if(layer <= OSD_LAYER3)
  700. ark1668_lcdc_set_osd_pos(layer, x, y);
  701. else
  702. ark1668_lcdc_set_video_layer_pos(layer-OSD_LAYER_MAX, x, y);
  703. printk(KERN_DEBUG "layer=%d: x=%d, y=%d.\n ",layer, x, y);
  704. break;
  705. }
  706. case VIN_SET_SOURCE_SIZE:
  707. {
  708. unsigned int width, height, data;
  709. data = *(unsigned int*)arg;
  710. width = data & 0xFFFF;
  711. height = (data >> 16) & 0xFFFF;
  712. if(layer <= OSD_LAYER3)
  713. ark1668_lcdc_set_osd_size(layer, width, height);
  714. else{
  715. ark1668_lcdc_set_video_source_size(layer-OSD_LAYER_MAX, width, height);
  716. ark1668_lcdc_set_video_win_size(layer-OSD_LAYER_MAX, width, height);
  717. ark1668_lcdc_set_video_win_point(layer-OSD_LAYER_MAX, 0, 0);
  718. ark1668_lcdc_set_video_layer_size(layer-OSD_LAYER_MAX, width, height);
  719. ark1668_lcdc_set_video_scal(layer-OSD_LAYER_MAX, width, height,
  720. 0, 0, 0, 0, width, height, 0, 0);
  721. }
  722. printk(KERN_DEBUG "layer=%d: width=%d, height=%d.\n ",layer, width, height);
  723. break;
  724. }
  725. case VIN_SET_LAYER_SIZE:
  726. {
  727. unsigned int width, height, data;
  728. data = *(unsigned int*)arg;
  729. width = data & 0xFFFF;
  730. height = (data >> 16) & 0xFFFF;
  731. ark1668_lcdc_set_video_layer_size(layer-OSD_LAYER_MAX, width, height);
  732. break;
  733. }
  734. case VIN_SET_WINDOW_FORMAT:
  735. {
  736. unsigned int data, format, yuv_order, rgb_order;
  737. data = *(unsigned int*)arg;
  738. format = (data >> 0) & 0xFF;
  739. yuv_order = (data >> 16) & 0xF;
  740. rgb_order = (data >> 24) & 0xF;
  741. if(layer <= OSD_LAYER3)
  742. ark1668_lcdc_set_osd_format(layer, format, yuv_order, rgb_order);
  743. else
  744. ark1668_lcdc_set_video_format(layer-OSD_LAYER_MAX, format, yuv_order, rgb_order, 0);
  745. if(layer == OSD_LAYER3){
  746. ark1668_lcdc_set_osd_alpha_blend_en_lcd(layer, 1);
  747. ark1668_lcdc_set_osd_per_pix_alpha_blend_en_lcd(layer, 1);
  748. }
  749. printk(KERN_DEBUG "layer=%d: format=%d.\n ",layer, format);
  750. break;
  751. }
  752. case VIN_SET_WINDOW_ADDR:
  753. {
  754. struct ark_disp_addr* addr = (struct ark_disp_addr*)arg;
  755. //printk(KERN_ALERT "VIN_SET_WINDOW_ADDR :layer=%d: yaddr=0x%0x, cbaddr=0x%0x, craddr=0x%0x.\n ",layer, addr->yaddr, addr->cbaddr, addr->craddr);
  756. if(layer <= OSD_LAYER3)
  757. ark1668_lcdc_set_osd_addr(layer, addr->yaddr);
  758. else
  759. ark1668_lcdc_set_video_addr(layer-OSD_LAYER_MAX, addr->yaddr, addr->cbaddr, addr->craddr);
  760. break;
  761. }
  762. case VIN_SET_WINDOW_SCALER:
  763. {
  764. struct ark_disp_scaler* scaler = (struct ark_disp_scaler*)arg;
  765. if(layer <= OSD_LAYER3){
  766. error = -EINVAL;
  767. printk("%s: layer<=OSD_LAYER3 can not scaler\n", __func__);
  768. goto end;
  769. }
  770. ark1668_lcdc_set_video_scal(layer-OSD_LAYER_MAX, scaler->src_w, scaler->src_h, 0, 0, 0, 0, scaler->out_w, scaler->out_h, 0, 0);
  771. printk(KERN_DEBUG "layer=%d: scaler src_w=%d, src_h=%d, out_w=%d, out_h=%d.\n ",layer, scaler->src_w, scaler->src_h, scaler->out_w, scaler->out_h);
  772. break;
  773. }
  774. default:
  775. printk("%s %d: unknown ioctl %08x\n",__FUNCTION__, __LINE__, cmd);
  776. break;
  777. }
  778. return 0;
  779. end:
  780. return error;
  781. }
  782. EXPORT_SYMBOL(ark168vin_set_display);
  783. void ark1668_lcdc_display_update_atomic(struct ark1668_lcdfb_info* sinfo)
  784. {
  785. unsigned int format, yuv_order, rgb_order, i, layer;
  786. struct ark_disp_atomic* p;
  787. if(!sinfo->atomic_flag)
  788. return;
  789. for(i = 0; i < ARK1668_LAYER_MAX;i++){
  790. if(!(sinfo->atomic_flag & (1 << i)))
  791. continue;
  792. p = &sinfo->patomic[i];
  793. if(!p->atomic_stat || p->layer < 0 || p->layer > 4){
  794. sinfo->atomic_flag &= ~(1 << i);
  795. memset(&sinfo->patomic[i], 0 ,sizeof(struct ark_disp_atomic));
  796. continue;
  797. }
  798. //printk(KERN_ALERT "%s: atomic_stat=0x%0x, layer=%d.\n ",__func__, p->atomic_stat, p->layer);
  799. if(p->layer >= OSD_LAYER1 && p->layer <= OSD_LAYER3){
  800. layer = p->layer;
  801. if(p->atomic_stat & ATOMIC_SET_LAYER_POS)
  802. ark1668_lcdc_set_osd_pos(layer, p->pos_x, p->pos_y);
  803. if(p->atomic_stat & ATOMIC_SET_LAYER_SIZE)
  804. ark1668_lcdc_set_osd_size(layer, p->width, p->height);
  805. if(p->atomic_stat & ATOMIC_SET_LAYER_FMT){
  806. format = (p->format >> 0) & 0xFF;
  807. yuv_order = (p->format >> 16) & 0xF;
  808. rgb_order = (p->format >> 24) & 0xF;
  809. ark1668_lcdc_set_osd_format(layer, format, yuv_order, rgb_order);
  810. if(format == ARK1668_LCDC_FORMAT_RGBA888){
  811. ark1668_lcdc_set_osd_alpha_blend_en_lcd(layer, 1);
  812. ark1668_lcdc_set_osd_per_pix_alpha_blend_en_lcd(layer, 1);
  813. }
  814. //printk(KERN_ALERT "%s: format=%d, yuv_order=%d, rgb_order=%d.\n ",__func__, format, yuv_order,rgb_order);
  815. }
  816. if(p->atomic_stat & ATOMIC_SET_LAYER_ADDR)
  817. ark1668_lcdc_set_osd_addr(layer, p->addr.yaddr);
  818. }else{
  819. layer = p->layer-OSD_LAYER_MAX;
  820. if(p->atomic_stat & ATOMIC_SET_LAYER_POS)
  821. ark1668_lcdc_set_video_layer_pos(layer, p->pos_x, p->pos_y);
  822. if(p->atomic_stat & ATOMIC_SET_LAYER_SIZE){
  823. ark1668_lcdc_set_video_source_size(layer, p->width, p->height);
  824. ark1668_lcdc_set_video_win_size(layer, p->width, p->height);
  825. ark1668_lcdc_set_video_win_point(layer, 0, 0);
  826. ark1668_lcdc_set_video_layer_size(layer, p->width, p->height);
  827. if(!(p->atomic_stat & ATOMIC_SET_LAYER_SCALER))
  828. ark1668_lcdc_set_video_scal(layer, p->width, p->height,
  829. 0, 0, 0, 0, p->width, p->height, 0, 0);
  830. }
  831. if(p->atomic_stat & ATOMIC_SET_LAYER_FMT){
  832. format = (p->format >> 0) & 0xFF;
  833. yuv_order = (p->format >> 16) & 0xF;
  834. rgb_order = (p->format >> 24) & 0xF;
  835. ark1668_lcdc_set_video_format(layer, format, yuv_order, rgb_order, 0);
  836. }
  837. if(p->atomic_stat & ATOMIC_SET_LAYER_ADDR)
  838. ark1668_lcdc_set_video_addr(layer, p->addr.yaddr, p->addr.cbaddr, p->addr.craddr);
  839. if(p->atomic_stat & ATOMIC_SET_LAYER_SCALER)
  840. ark1668_lcdc_set_video_scal(layer, p->scaler.src_w, p->scaler.src_h,
  841. 0, 0, 0, 0, p->scaler.out_w, p->scaler.out_h, 0, 0);
  842. }
  843. sinfo->atomic_flag &= ~(1 << i);
  844. memset(&sinfo->patomic[i], 0 ,sizeof(struct ark_disp_atomic));
  845. }
  846. }
  847. int ark1668_lcdc_wait_for_vsync(void)
  848. {
  849. struct ark1668_lcdfb_info *sinfo = lcdfb_info;
  850. int ret;
  851. if(!sinfo) return -EINVAL;
  852. sinfo->vsync_flag = 0;
  853. ret = wait_event_interruptible_timeout(sinfo->vsync_waitq,
  854. sinfo->vsync_flag != 0,
  855. msecs_to_jiffies(100)); // 100ms at most
  856. if (ret < 0)
  857. return ret;
  858. if (ret == 0)
  859. return -ETIMEDOUT;
  860. if(sinfo->atomic_flag)
  861. ark1668_lcdc_display_update_atomic(sinfo);
  862. return 0;
  863. }
  864. EXPORT_SYMBOL(ark1668_lcdc_wait_for_vsync);
  865. int ark1668_lcdfb_ioctl(struct fb_info *info, unsigned int cmd, unsigned long arg)
  866. {
  867. struct ark1668_lcdfb_info *sinfo = info->par;
  868. unsigned long error = 0;
  869. int layer = info->node;
  870. switch (cmd) {
  871. case FBIO_WAITFORVSYNC:
  872. case ARKFB_WAITFORVSYNC:
  873. ark1668_lcdc_wait_for_vsync();
  874. //printk(KERN_DEBUG "ark1668 lcdc wait for vsync.\n ");
  875. break;
  876. case ARKFB_SHOW_WINDOW:
  877. if(layer <= OSD_LAYER3)
  878. ark1668_lcdc_set_osd_en(layer, 1);
  879. else
  880. ark1668_lcdc_set_video_en(layer-OSD_LAYER_MAX, 1);
  881. printk(KERN_DEBUG "layer=%d: show window.\n ",layer);
  882. break;
  883. case ARKFB_HIDE_WINDOW:
  884. if(layer <= OSD_LAYER3)
  885. ark1668_lcdc_set_osd_en(layer, 0);
  886. else
  887. ark1668_lcdc_set_video_en(layer-OSD_LAYER_MAX, 0);
  888. printk(KERN_DEBUG "layer=%d: hide window.\n ",layer);
  889. break;
  890. case ARKFB_SET_WINDOW_POS:
  891. {
  892. unsigned int x,y,data;
  893. if(copy_from_user(&data, (void *)arg, sizeof(unsigned int))){
  894. printk("%s: copy from user para error\n", __func__);
  895. error = -EFAULT;
  896. goto end;
  897. }
  898. x = data & 0xFFFF;
  899. y = (data >> 16) & 0xFFFF;
  900. if(layer <= OSD_LAYER3)
  901. ark1668_lcdc_set_osd_pos(layer, x, y);
  902. else
  903. ark1668_lcdc_set_video_layer_pos(layer-OSD_LAYER_MAX, x, y);
  904. printk(KERN_DEBUG "layer=%d: x=%d, y=%d.\n ",layer, x, y);
  905. }
  906. break;
  907. case ARKFB_SET_WINDOW_SIZE:
  908. {
  909. unsigned int width, height, data;
  910. if(copy_from_user(&data, (void *)arg, sizeof(unsigned int))){
  911. printk("%s: copy from user para error\n", __func__);
  912. error = -EFAULT;
  913. goto end;
  914. }
  915. width = data & 0xFFFF;
  916. height = (data >> 16) & 0xFFFF;
  917. if(layer <= OSD_LAYER3)
  918. ark1668_lcdc_set_osd_size(layer, width, height);
  919. else{
  920. ark1668_lcdc_set_video_source_size(layer-OSD_LAYER_MAX, width, height);
  921. ark1668_lcdc_set_video_win_size(layer-OSD_LAYER_MAX, width, height);
  922. ark1668_lcdc_set_video_win_point(layer-OSD_LAYER_MAX, 0, 0);
  923. ark1668_lcdc_set_video_layer_size(layer-OSD_LAYER_MAX, width, height);
  924. ark1668_lcdc_set_video_scal(layer-OSD_LAYER_MAX, width, height,
  925. 0, 0, 0, 0, width, height, 0, 0);
  926. }
  927. printk(KERN_DEBUG "layer=%d: width=%d, height=%d.\n ",layer, width, height);
  928. }
  929. break;
  930. case ARKFB_SET_WINDOW_FORMAT:
  931. {
  932. unsigned int data, format, yuv_order, rgb_order;
  933. if(copy_from_user(&data, (void *)arg, sizeof(unsigned int))){
  934. printk("%s: copy from user para error\n", __func__);
  935. error = -EFAULT;
  936. goto end;
  937. }
  938. format = (data >> 0) & 0xFF;
  939. yuv_order = (data >> 16) & 0xF;
  940. rgb_order = (data >> 24) & 0xF;
  941. if(layer <= OSD_LAYER3)
  942. ark1668_lcdc_set_osd_format(layer, format, yuv_order, rgb_order);
  943. else
  944. ark1668_lcdc_set_video_format(layer-OSD_LAYER_MAX, format, yuv_order, rgb_order, 0);
  945. if(layer == OSD_LAYER3){
  946. ark1668_lcdc_set_osd_alpha_blend_en_lcd(layer, 1);
  947. ark1668_lcdc_set_osd_per_pix_alpha_blend_en_lcd(layer, 1);
  948. }
  949. printk(KERN_DEBUG "layer=%d: format=%d.\n ",layer, format);
  950. }
  951. break;
  952. case ARKFB_SET_WINDOW_ADDR:
  953. {
  954. struct ark_disp_addr addr;
  955. if(copy_from_user(&addr, (void *)arg, sizeof(struct ark_disp_addr))){
  956. printk("%s: copy from user para error\n", __func__);
  957. error = -EFAULT;
  958. goto end;
  959. }
  960. memcpy(&sinfo->render_addr[layer], &addr, sizeof(struct ark_disp_addr));
  961. //printk(KERN_ALERT "layer=%d: yaddr=0x%0x, cbaddr=0x%0x, craddr=0x%0x.\n ",layer, addr.yaddr, addr.cbaddr, addr.craddr);
  962. }
  963. break;
  964. case ARKFB_SET_WINDOW_SCALER:
  965. {
  966. struct ark_disp_scaler scaler;
  967. if(layer <= OSD_LAYER3){
  968. error = -EINVAL;
  969. printk("%s: layer<=OSD_LAYER3 can not scaler\n", __func__);
  970. goto end;
  971. }
  972. if(copy_from_user(&scaler, (void *)arg, sizeof(struct ark_disp_scaler))){
  973. printk("%s: copy from user para error\n", __func__);
  974. error = -EFAULT;
  975. goto end;
  976. }
  977. ark1668_lcdc_set_video_scal(layer-OSD_LAYER_MAX, scaler.src_w, scaler.src_h, 0, 0, 0, 0, scaler.out_w, scaler.out_h, 0, 0);
  978. printk(KERN_DEBUG "layer=%d: scaler src_w=%d, src_h=%d, out_w=%d, out_h=%d.\n ",layer, scaler.src_w, scaler.src_h, scaler.out_w, scaler.out_h);
  979. }
  980. break;
  981. case ARKFB_SET_WINDOW_PRIORITY:
  982. {
  983. unsigned int layer_id[5];
  984. if(copy_from_user(&layer_id, (void*)arg, sizeof(unsigned int)*5)){
  985. printk(KERN_ALERT "ARKFB_SET_WINDOW_PRIORITY error\n");
  986. return -EFAULT;
  987. }
  988. printk(KERN_ALERT "++++++ARKFB_SET_WINDOW_PRIORITY layer_id: %d.%d,%d,%d,%d\n",layer_id[0],layer_id[1],layer_id[2],layer_id[3],layer_id[4]);
  989. ark1668_lcdc_set_video_priority(layer_id[0]);
  990. ark1668_lcdc_set_video2_priority(layer_id[1]);
  991. ark1668_lcdc_set_win1_priority(layer_id[2]);
  992. ark1668_lcdc_set_win2_priority(layer_id[3]);
  993. ark1668_lcdc_set_win3_priority(layer_id[4]);
  994. }
  995. break;
  996. case ARKFB_SET_WINDOW_ATOMIC:
  997. {
  998. struct ark_disp_atomic atomic;
  999. if(copy_from_user(&atomic, (void *)arg, sizeof(struct ark_disp_atomic))){
  1000. printk("%s: copy from user para error\n", __func__);
  1001. error = -EFAULT;
  1002. goto end;
  1003. }
  1004. if(!atomic.atomic_stat || atomic.layer != layer){
  1005. printk("%s: atomic_stat or layer error\n", __func__);
  1006. error = -EFAULT;
  1007. goto end;
  1008. }
  1009. printk(KERN_DEBUG "%s===>layer=%d, atomic_stat=0x%0x.\n ",__func__, layer, atomic.atomic_stat);
  1010. sinfo->atomic_flag |= (1 << layer);
  1011. memcpy(&sinfo->patomic[layer], &atomic, sizeof(struct ark_disp_atomic));
  1012. ark1668_lcdc_wait_for_vsync();
  1013. }
  1014. break;
  1015. case ARKFB_SET_REG_VALUE:
  1016. {
  1017. struct ark_disp_reg reg;
  1018. if(copy_from_user(&reg, (void *)arg, sizeof(struct ark_disp_reg))){
  1019. printk("%s: copy from user para error\n", __func__);
  1020. error = -EFAULT;
  1021. goto end;
  1022. }
  1023. if((reg.addr & 0xffff0000) == 0xe0500000){
  1024. writel(reg.value, lcdc_base + (reg.addr&0xffff));
  1025. printk("arkfb write reg:0x%0x=0x%0x.\n ", reg.addr, reg.value);
  1026. }else{
  1027. error = -EINVAL;
  1028. goto end;
  1029. }
  1030. }
  1031. break;
  1032. case ARKFB_GET_REG_VALUE:
  1033. {
  1034. struct ark_disp_reg reg;
  1035. if(copy_from_user(&reg, (void *)arg, sizeof(struct ark_disp_reg))){
  1036. printk("%s: copy from user para error\n", __func__);
  1037. error = -EFAULT;
  1038. goto end;
  1039. }
  1040. if((reg.addr & 0xffff0000) == 0xe0500000){
  1041. reg.value = readl(lcdc_base + (reg.addr&0xffff));
  1042. printk("arkfb read reg:0x%0x=0x%0x.\n ", reg.addr, reg.value);
  1043. }else{
  1044. error = -EINVAL;
  1045. goto end;
  1046. }
  1047. if(copy_to_user((void *)arg, &reg, sizeof(struct ark_disp_reg))){
  1048. printk("%s: copy to user para error\n", __func__);
  1049. error = -EFAULT;
  1050. goto end;
  1051. }
  1052. }
  1053. break;
  1054. case ARKFB_GET_WINDOW_ADDR:
  1055. {
  1056. struct ark_disp_addr addr;
  1057. memset(&addr, 0, sizeof(struct ark_disp_addr));
  1058. if(layer <= OSD_LAYER3)
  1059. addr.yaddr = ark1668_lcdc_get_osd_addr(layer);
  1060. else
  1061. ark1668_lcdc_get_video_addr(layer-OSD_LAYER_MAX, &addr.yaddr, &addr.cbaddr, &addr.craddr);
  1062. if(copy_to_user((void *)arg, &addr, sizeof(struct ark_disp_addr))){
  1063. printk("%s: copy to user para error\n", __func__);
  1064. error = -EFAULT;
  1065. goto end;
  1066. }
  1067. //printk(KERN_DEBUG "%s: layer=%d: yaddr=0x%0x, cbaddr=0x%0x, craddr=0x%0x.\n ",__func__, layer, addr.yaddr, addr.cbaddr, addr.craddr);
  1068. }
  1069. break;
  1070. case ARKFB_GET_SCREEN_INFO:
  1071. {
  1072. struct ark_screen screen;
  1073. memset(&screen, 0, sizeof(struct ark_screen));
  1074. screen.width = screen.disp_width = lcdc_width;
  1075. screen.height = screen.disp_height = lcdc_height;
  1076. if(copy_to_user((void *)arg, &screen, sizeof(struct ark_screen))){
  1077. printk("%s: copy to user para error\n", __func__);
  1078. error = -EFAULT;
  1079. goto end;
  1080. }
  1081. //printk(KERN_ALERT "%s: type=%d, width=%d, height=%d.\n ", __func__, screen.type, screen.width,screen.height);
  1082. }
  1083. break;
  1084. case ARKFB_SET_SCREEN_INFO:
  1085. {
  1086. struct ark_screen screen;
  1087. if(copy_from_user(&screen, (void *)arg, sizeof(struct ark_screen))){
  1088. printk("%s: copy to user para error\n", __func__);
  1089. error = -EFAULT;
  1090. goto end;
  1091. }
  1092. ///////////////// Reserved///////////////////
  1093. }
  1094. break;
  1095. case ARKFB_GET_PLATFORM_INFO:
  1096. {
  1097. struct ark_platform_info platform;
  1098. memset(&platform, 0, sizeof(struct ark_platform_info));
  1099. platform.type = ARK_PLATFORM_ARK1668;
  1100. if(copy_to_user((void *)arg, &platform, sizeof(struct ark_platform_info))){
  1101. printk("%s: copy to user para error\n", __func__);
  1102. error = -EFAULT;
  1103. goto end;
  1104. }
  1105. }
  1106. break;
  1107. default:
  1108. printk("%s %d: unknown ioctl %08x\n",__FUNCTION__, __LINE__, cmd);
  1109. break;
  1110. }
  1111. end:
  1112. return error;
  1113. }