platform.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: platform.c 593 2006-01-17 22:25:08Z picard $
  18.  *
  19.  * The Core Pocket Media Player
  20.  * Copyright (c) 2004-2005 Gabor Kovacs
  21.  *
  22.  ****************************************************************************/
  23. #include "common.h"
  24. #include "cpu/cpu.h"
  25. int DebugMask() 
  26. {
  27. return DEBUG_SYS;//|DEBUG_FORMAT|DEBUG_PLAYER;//|DEBUG_VIDEO;//|DEBUG_AUDIO|DEBUG_VIDEO|DEBUG_PLAYER;//|DEBUG_VIDEO;//DEBUG_TEST;//|DEBUG_FORMAT|DEBUG_PLAYER;
  28. }
  29. static const datatable Params[] = 
  30. {
  31. { PLATFORM_LANG, TYPE_INT, DF_SETUP|DF_ENUMSTRING|DF_RESTART, LANG_ID },
  32. { PLATFORM_TYPE, TYPE_STRING, DF_SETUP|DF_RDONLY|DF_GAP },
  33. { PLATFORM_VER, TYPE_INT, DF_SETUP|DF_RDONLY|DF_HIDDEN },
  34. { PLATFORM_VERSION, TYPE_STRING, DF_SETUP|DF_RDONLY },
  35. { PLATFORM_OEMINFO, TYPE_STRING, DF_SETUP|DF_RDONLY },
  36. { PLATFORM_TYPENO, TYPE_INT, DF_SETUP|DF_RDONLY|DF_HIDDEN },
  37. { PLATFORM_MODEL, TYPE_INT, DF_SETUP|DF_RDONLY|DF_HIDDEN },
  38. { PLATFORM_CAPS, TYPE_INT, DF_SETUP|DF_RDONLY|DF_HIDDEN|DF_HEX },
  39. { PLATFORM_CPU, TYPE_STRING, DF_SETUP|DF_RDONLY },
  40. { PLATFORM_CPUMHZ, TYPE_INT, DF_SETUP|DF_RDONLY|DF_MHZ },
  41. { PLATFORM_ICACHE, TYPE_INT, DF_SETUP|DF_RDONLY|DF_HIDDEN },
  42. { PLATFORM_DCACHE, TYPE_INT, DF_SETUP|DF_RDONLY|DF_HIDDEN },
  43. { PLATFORM_WMPVERSION, TYPE_INT, DF_SETUP|DF_RDONLY|DF_HIDDEN },
  44. { PLATFORM_ORIENTATION, TYPE_INT, DF_SETUP|DF_RDONLY|DF_HIDDEN },
  45. DATATABLE_END(PLATFORM_ID)
  46. };
  47. #if defined(TARGET_SYMBIAN) && !defined(ARM)
  48. static INLINE int64_t STDCALL CPUSpeedClk(int i) { return 0; }
  49. #else
  50. extern int64_t STDCALL CPUSpeedClk(int);
  51. #endif
  52. int CPUSpeed()
  53. {
  54. int Speed = 0;
  55. #if defined(MIPS) || defined(ARM) || defined(_M_IX86) || defined(SH3)
  56. int Old = ThreadPriority(NULL,-100);
  57. int TimeFreq = GetTimeFreq();
  58. if (TimeFreq>0)
  59. {
  60. #ifdef TARGET_PALMOS
  61. int Iter = 1;
  62. #else
  63. int Iter = 4;
  64. #endif
  65. int64_t Clk,Best = 0;
  66. int BestCount = 0;
  67. int n,Count;
  68. int Sub=1,SubTime=8;
  69. int c0[2],c1[2],c2[2];
  70. int t;
  71. int SubTimeLimit = TimeFreq/250;
  72. if (SubTimeLimit <= 1)
  73. SubTimeLimit = 2;
  74. Count = 800000 / TimeFreq;
  75. for (n=0;n<4*Iter;++n)
  76. {
  77. GetTimeCycle(c0);
  78. GetTimeCycle(c1);
  79. GetTimeCycle(c2);
  80. c2[0] -= c1[0];
  81. c1[0] -= c0[0];
  82. if (c2[1]<c1[1])
  83. {
  84. c1[0] = c2[0];
  85. c1[1] = c2[1];
  86. }
  87. if (c1[1]*SubTime > c1[0]*Sub)
  88. {
  89. Sub = c1[1];
  90. SubTime = c1[0];
  91. }
  92. if (n>6 && SubTime >= SubTimeLimit)
  93. break;
  94. }
  95. if (SubTime < SubTimeLimit)
  96. {
  97. // Sub = approx number of possbile GetTime() calls per SubTime ms
  98. for (n=0;n<16*Iter;++n)
  99. {
  100. CPUSpeedClk(1); // load instruction cache
  101. GetTimeCycle(c0);
  102. Clk = CPUSpeedClk(Count);
  103. GetTimeCycle(c1);
  104. t = c1[0]-c0[0];
  105. #if defined(TARGET_WINCE)
  106. if (QueryPlatform(PLATFORM_TYPENO) == TYPE_SMARTPHONE)
  107. Clk += t*2000;
  108. #elif defined(TARGET_WIN32)
  109. // we are assuming a 1ms timer interrupt with 
  110. // minimum 6500 cycles interrupt handler
  111. Clk += t*6500;
  112. #endif
  113. t *= 100000;
  114. t -=(c1[1] * 100000 * SubTime)/Sub; // adjust with subtime
  115. if (t < 25000) // quater tick -> double count
  116. Count *= 2;
  117. else
  118. if (t < 50000) // we need at least half tick
  119. Count += Count/8;
  120. else
  121. {
  122. #ifdef ARM
  123. Clk = (Clk *  99600 * TimeFreq) / t;
  124. #else
  125. Clk = (Clk * 100000 * TimeFreq) / t;
  126. #endif
  127. if (Best < Clk)
  128. {
  129. Best = Clk;
  130. BestCount = 1;
  131. }
  132. else
  133. if ((Best * 127)/128 < Clk && ++BestCount>4*Iter)
  134. break;
  135. }
  136. }
  137. }
  138. Speed = (int)((Best + 500000)/1000000);
  139. }
  140. //rounding
  141. if ((Speed % 104)==1 || Speed==207) --Speed;
  142. else if ((Speed % 104)==103 || Speed==205) ++Speed;
  143. else if ((Speed % 10)==1) --Speed;
  144. else if ((Speed % 10)==9) ++Speed;
  145. else if (((Speed % 100)%33)==1) --Speed;
  146. else if (((Speed % 100)%33)==32) ++Speed;
  147. ThreadPriority(NULL,Old);
  148. #endif
  149. return Speed;
  150. }
  151. static int Enum(platform* p, int* No, datadef* Param)
  152. {
  153. return NodeEnumTable(No,Param,Params);
  154. }
  155. static int Get(platform* p, int No, void* Data, int Size)
  156. {
  157. int Result = ERR_INVALID_PARAM;
  158. switch (No)
  159. {
  160. case PLATFORM_LANG: GETVALUE(Context()->Lang,int); break;
  161. case PLATFORM_TYPE: GETSTRING(p->PlatformType); break;
  162. case PLATFORM_WMPVERSION: GETVALUE(p->WMPVersion,int); break;
  163. case PLATFORM_ORIENTATION: GETVALUE(GetOrientation(),int); break;
  164. case PLATFORM_CPU: GETSTRING(p->CPU); break;
  165. case PLATFORM_OEMINFO: if (p->OemInfo[0]) GETSTRING(p->OemInfo); break;
  166. case PLATFORM_VER: GETVALUE(p->Ver,int); break;
  167. case PLATFORM_TYPENO: GETVALUE(p->Type,int); break;
  168. case PLATFORM_MODEL: GETVALUE(p->Model,int); break;
  169. case PLATFORM_CAPS: GETVALUE(p->Caps,int); break;
  170. case PLATFORM_ICACHE: GETVALUECOND(p->ICache,int,p->ICache>0); break;
  171. case PLATFORM_DCACHE: GETVALUECOND(p->DCache,int,p->DCache>0); break;
  172. case PLATFORM_VERSION: GETSTRING(p->Version); break;
  173. case PLATFORM_CPUMHZ: 
  174. assert(Size==sizeof(int));
  175. if ((*(int*)Data = CPUSpeed())>0)
  176. Result = ERR_NONE;
  177. break;
  178. }
  179. return Result;
  180. }
  181. static int Set(platform* p, int No, const void* Data, int Size)
  182. {
  183. int Result = ERR_INVALID_PARAM;
  184. switch (No)
  185. {
  186. case PLATFORM_LANG: SETVALUE(Context()->Lang,int,ERR_NONE); break;
  187. }
  188. return Result;
  189. }
  190. int QueryPlatform(int Param)
  191. {
  192. node* Platform = Context()->Platform;
  193. int Value = 0;
  194. if (Platform)
  195. Platform->Get(Platform,Param,&Value,sizeof(Value));
  196. return Value;
  197. }
  198. extern void PlatformDetect(platform*);
  199. static int Create(platform* p)
  200. {
  201. cpudetect Info;
  202. p->Node.Enum = (nodeenum)Enum;
  203. p->Node.Get = (nodeget)Get;
  204. p->Node.Set = (nodeset)Set;
  205. p->Model = MODEL_UNKNOWN;
  206. CPUDetect(&Info);
  207. p->Caps = Info.Caps;
  208. p->ICache = Info.ICache;
  209. p->DCache = Info.DCache;
  210. p->CPU[0] = 0;
  211. if (Info.Arch)
  212. tcscat_s(p->CPU,TSIZEOF(p->CPU),Info.Arch);
  213. if (Info.Vendor)
  214. stcatprintf_s(p->CPU,TSIZEOF(p->CPU),T(" %s"),Info.Vendor);
  215. if (Info.Model)
  216. stcatprintf_s(p->CPU,TSIZEOF(p->CPU),T(" %s"),Info.Model);
  217. PlatformDetect(p);
  218. return ERR_NONE;
  219. }
  220. const nodedef Platform = 
  221. {
  222. sizeof(platform)|CF_GLOBAL|CF_SETTINGS,
  223. PLATFORM_ID,
  224. NODE_CLASS,
  225. PRI_MAXIMUM+650,
  226. (nodecreate)Create,
  227. NULL,
  228. };