overlay_s1d13806.c
上传用户:wstnjxml
上传日期:2014-04-03
资源大小:7248k
文件大小:7k
源码类别:

Windows CE

开发平台:

C/C++

  1. /*****************************************************************************
  2.  *
  3.  * This program is free software ; you can redistribute it and/or modify
  4.  * it under the terms of the GNU General Public License as published by
  5.  * the Free Software Foundation; either version 2 of the License, or
  6.  * (at your option) any later version.
  7.  *
  8.  * This program is distributed in the hope that it will be useful,
  9.  * but WITHOUT ANY WARRANTY; without even the implied warranty of
  10.  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  11.  * GNU General Public License for more details.
  12.  *
  13.  * You should have received a copy of the GNU General Public License
  14.  * along with this program; if not, write to the Free Software
  15.  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307 USA
  16.  *
  17.  * $Id: overlay_s1d13806.c 543 2006-01-07 22:06:24Z picard $
  18.  *
  19.  * The Core Pocket Media Player
  20.  * Copyright (c) 2004-2005 Gabor Kovacs
  21.  *
  22.  ****************************************************************************/
  23. #include "../common.h"
  24. #if defined(TARGET_WINCE) && defined(ARM)
  25. #define WIN32_LEAN_AND_MEAN
  26. #ifndef STRICT
  27. #define STRICT
  28. #endif
  29. #include <windows.h>
  30. extern int STDCALL IRQEnable();
  31. extern int STDCALL IRQDisable();
  32. typedef struct s1d13806
  33. {
  34. overlay Overlay;
  35. planes Planes;
  36. bool_t KernelMode;
  37. void* Buffer;
  38. bool_t PhyAddr;
  39. volatile unsigned char* Regs;
  40. volatile unsigned short* FIFO;
  41. HBRUSH Brush;
  42. } s1d13806;
  43. static INLINE void Reg3(s1d13806* p,int i,int v)
  44. {
  45. p->Regs[i+0] = (unsigned char)(v);
  46. p->Regs[i+1] = (unsigned char)(v >> 8);
  47. p->Regs[i+2] = (unsigned char)(v >> 16);
  48. }
  49. static INLINE void Reg2(s1d13806* p,int i,int v)
  50. {
  51. p->Regs[i+0] = (unsigned char)(v);
  52. p->Regs[i+1] = (unsigned char)(v >> 8);
  53. }
  54. static int Blit(overlay* p, const constplanes Data, const constplanes DataLast)
  55. {
  56. planes Planes;
  57. int Result = p->Lock(p,Planes,1);
  58. if (Result==ERR_NONE)
  59. {
  60. BlitImage(p->Soft,Planes,Data,DataLast,-1,-1);
  61. p->Unlock(p);
  62. }
  63. return Result;
  64. }
  65. static int Update(overlay* p)
  66. {
  67. blitfx FX = p->FX;
  68. video Output = p->Output.Format.Video;
  69. rect OldGUI = p->GUIAlignedRect;
  70. rect Old = p->DstAlignedRect;
  71. VirtToPhy(&p->Viewport,&p->DstAlignedRect,&p->Output.Format.Video);
  72. VirtToPhy(NULL,&p->SrcAlignedRect,&p->Input.Format.Video);
  73. p->DstAlignedRect.Height >>= 1;
  74. p->DstAlignedRect.y >>= 1;
  75. Output.Height >>= 1;
  76. Output.Pitch <<= 1;
  77. if (FX.Direction & DIR_SWAPXY)
  78. FX.ScaleX >>= 1;
  79. else
  80. FX.ScaleY >>= 1;
  81. if (p->Output.Format.Video.Pixel.BitCount!=16)
  82. FX.Flags |= BLITFX_DITHER;
  83. BlitRelease(p->Soft);
  84. p->Soft = BlitCreate(&Output,&p->Input.Format.Video,&FX,&p->Caps);
  85. BlitAlign(p->Soft,&p->DstAlignedRect, &p->SrcAlignedRect);
  86. p->DstAlignedRect.Height <<= 1;
  87. p->DstAlignedRect.y <<= 1;
  88. if (p->Output.Format.Video.Pixel.BitCount!=16)
  89. p->Caps &= ~VC_DITHER;
  90. PhyToVirt(&p->DstAlignedRect,&p->GUIAlignedRect,&p->Output.Format.Video);
  91. if (!EqRect(&Old,&p->DstAlignedRect) && p->Show && p->Primary)
  92. {
  93. WinInvalidate(&OldGUI,0);
  94. WinInvalidate(&p->Viewport,1);
  95. WinValidate(&p->GUIAlignedRect);
  96. }
  97. if (p->Show && (p->FullScreenViewport || !p->Primary))
  98. OverlayClearBorder(p);
  99. return ERR_NONE;
  100. }
  101. static int Init(s1d13806* p)
  102. {
  103. HKEY Key;
  104. video* Output = &p->Overlay.Output.Format.Video;
  105. QueryDesktop(Output);
  106. DefaultPitch(Output);
  107. p->Regs = NULL;
  108. p->Buffer = NULL;
  109. if (RegOpenKeyEx(HKEY_LOCAL_MACHINE, T("Drivers\Display\S1D13806"), 0, KEY_READ, &Key) == ERROR_SUCCESS)
  110. {
  111. DWORD Value;
  112. DWORD RegSize;
  113. DWORD RegType;
  114. RegSize = sizeof(Value);
  115. if (RegQueryValueEx(Key, T("RegBase"), 0, &RegType, (LPBYTE)&Value, &RegSize) == ERROR_SUCCESS)
  116. p->Regs = (unsigned char*)Value;
  117. RegSize = sizeof(Value);
  118. if (RegQueryValueEx(Key, T("MemBase"), 0, &RegType, (LPBYTE)&Value, &RegSize) == ERROR_SUCCESS)
  119. p->Buffer = (void*)Value;
  120. RegCloseKey(Key);
  121. }
  122. if (!p->Regs || !p->Buffer)
  123. return ERR_NOT_SUPPORTED;
  124. p->PhyAddr = (uint32_t)p->Regs < 0x80000000;
  125. if (p->PhyAddr)
  126. {
  127. p->Regs = PhyMemBegin((uint32_t)p->Regs,2*1024*1024,0);
  128. if (!p->Regs)
  129. return ERR_NOT_SUPPORTED;
  130. p->Buffer = PhyMemBegin((uint32_t)p->Buffer,2*1024*1024,1);
  131. if (!p->Buffer)
  132. {
  133. PhyMemEnd((void*)p->Regs);
  134. p->Regs = NULL;
  135. return ERR_NOT_SUPPORTED;
  136. }
  137. }
  138. p->FIFO = (unsigned short*)(p->Regs+1024*1024);
  139. if (((int)p->Buffer & 15)==0 && (Output->Pitch & 15)==0)
  140. Output->Pixel.Flags |= PF_16ALIGNED;
  141. p->Planes[0] = p->Buffer;
  142. AdjustOrientation(Output,0);
  143. p->Overlay.SetFX = BLITFX_AVOIDTEARING;
  144. p->Brush = (HBRUSH)GetStockObject(BLACK_BRUSH);
  145. return ERR_NONE;
  146. }
  147. static void Done(s1d13806* p)
  148. {
  149. if (p->PhyAddr)
  150. {
  151. if (p->Buffer)
  152. PhyMemEnd(p->Buffer);
  153. if (p->Regs)
  154. PhyMemEnd((void*)p->Regs);
  155. }
  156. p->Buffer = NULL;
  157. p->Regs = NULL;
  158. }
  159. static int Reset(s1d13806* p)
  160. {
  161. Done(p);
  162. Init(p);
  163. return ERR_NONE;
  164. }
  165. static int Lock(s1d13806* p, planes Planes, bool_t OnlyAligned)
  166. {
  167. p->KernelMode = KernelMode(1);
  168. Planes[0] = p->Planes[0];
  169. return ERR_NONE;
  170. }
  171. static int Unlock(s1d13806* p)
  172. {
  173. if (p->Regs && p->Overlay.DstAlignedRect.Width>0 && p->Overlay.DstAlignedRect.Height>1)
  174. {
  175. int ReTry;
  176. HDC DC = GetDC(NULL);
  177. for (ReTry=0;ReTry<32;++ReTry)
  178. {
  179. // we have to make sure not to interfere with the GDI driver
  180. // use FillRect to make the GDI driver use the epson chip
  181. // disable the interrupt and verify we still in our FillRect blitting
  182. // wait until it's finished and start our own blitting
  183. RECT R;
  184. R.left = 0;
  185. R.top = 0;
  186. R.right = 1;
  187. R.bottom = p->Overlay.Output.Format.Video.Height-1;
  188. FillRect(DC,&R,p->Brush);
  189. IRQDisable();
  190. if ((p->Regs[0x100] & 128)==0 ||
  191. p->Regs[0x110] != 0 || p->Regs[0x111] != 0 ||
  192. p->Regs[0x112] != ((R.bottom-1) & 255) || p->Regs[0x113] != ((R.bottom-1) >> 8))
  193. {
  194. IRQEnable();
  195. }
  196. else
  197. {
  198. int Ofs = p->Overlay.DstAlignedRect.x*(p->Overlay.Output.Format.Video.Pixel.BitCount>>3) + 
  199. p->Overlay.DstAlignedRect.y*p->Overlay.Output.Format.Video.Pitch;
  200. while ((p->Regs[0x100] & 128)!=0);
  201. ReTry += p->FIFO[0];
  202. Reg3(p,0x104,Ofs);
  203. Reg3(p,0x108,Ofs + p->Overlay.Output.Format.Video.Pitch);
  204. Reg2(p,0x110,p->Overlay.DstAlignedRect.Width-1);
  205. Reg2(p,0x112,(p->Overlay.DstAlignedRect.Height/2)-1);
  206. p->Regs[0x103] = 2;
  207. p->Regs[0x102] = 0x0C;
  208. p->Regs[0x101] = (char)(p->Overlay.Output.Format.Video.Pixel.BitCount==16?1:0);
  209. Reg2(p,0x10C,p->Overlay.Output.Format.Video.Pitch);
  210. p->Regs[0x100] = 0x80;
  211. while ((p->Regs[0x100] & 128)==0);
  212. IRQEnable();
  213. break;
  214. }
  215. }
  216. ReleaseDC(NULL,DC);
  217. }
  218. KernelMode(p->KernelMode);
  219. return ERR_NONE;
  220. }
  221. static int Create(s1d13806* p)
  222. {
  223. if (Init(p) != ERR_NONE) // check if supported
  224. return ERR_NOT_SUPPORTED;
  225. Done(p);
  226. p->Overlay.Init = (ovlfunc)Init;
  227. p->Overlay.Done = (ovldone)Done;
  228. p->Overlay.Reset = (ovlfunc)Reset;
  229. p->Overlay.Lock = (ovllock)Lock;
  230. p->Overlay.Unlock = (ovlfunc)Unlock;
  231. p->Overlay.Blit = (ovlblit)Blit;
  232. p->Overlay.Update = (ovlfunc)Update;
  233. return ERR_NONE;
  234. }
  235. static const nodedef S1D13806 = 
  236. {
  237. sizeof(s1d13806)|CF_GLOBAL,
  238. S1D13806_ID,
  239. OVERLAY_CLASS,
  240. PRI_DEFAULT+92,
  241. (nodecreate)Create,
  242. };
  243. void OverlayS1D13806_Init()
  244. NodeRegisterClass(&S1D13806);
  245. }
  246. void OverlayS1D13806_Done()
  247. {
  248. NodeUnRegisterClass(S1D13806_ID);
  249. }
  250. #else
  251. void OverlayS1D13806_Init() {}
  252. void OverlayS1D13806_Done() {}
  253. #endif