FastTimes.c
上传用户:sun1608
上传日期:2007-02-02
资源大小:6116k
文件大小:12k
源码类别:

流媒体/Mpeg4/MP4

开发平台:

Visual C++

  1. /* File "FastTimes.c" - Original code by Matt Slot <fprefect@ambrosiasw.com>  */
  2. /* Created 4/24/99    - This file is hereby placed in the public domain       */
  3. /* Updated 5/21/99    - Calibrate to VIA, add TBR support, renamed functions  */
  4. /* Updated 10/4/99    - Use AbsoluteToNanoseconds() in case Absolute = double */
  5. /* Updated 2/15/00    - Check for native Time Manager, no need to calibrate   */
  6. /* Updated 2/19/00    - Fixed default value for gScale under native Time Mgr  */
  7. /* Updated 3/21/00    - Fixed ns conversion, create 2 different scale factors */
  8. /* Updated 5/03/00    - Added copyright and placed into PD. No code changes   */
  9. /* Updated 8/01/00    - Made "Carbon-compatible" by replacing LMGetTicks()    */
  10. /* This file is Copyright (C) Matt Slot, 1999-2000. It is hereby placed into 
  11.    the public domain. The author makes no warranty as to fitness or stability */
  12. #include <Gestalt.h>
  13. #include <LowMem.h>
  14. #include <CodeFragments.h>
  15. #include <DriverServices.h>
  16. #include <Timer.h>
  17. #include "FastTimes.h"
  18. /* **** **** **** **** **** **** **** **** **** **** **** **** **** **** **** */
  19. /* **** **** **** **** **** **** **** **** **** **** **** **** **** **** **** */
  20. /*
  21. On 680x0 machines, we just use Microseconds().
  22. On PowerPC machines, we try several methods:
  23.   * DriverServicesLib is available on all PCI PowerMacs, and perhaps
  24.     some NuBus PowerMacs. If it is, we use UpTime() : Overhead = 2.1 祍ec.
  25.   * The PowerPC 601 has a built-in "real time clock" RTC, and we fall
  26.     back to that, accessing it directly from asm. Overhead = 1.3 祍ec.
  27.   * Later PowerPCs have an accurate "time base register" TBR, and we 
  28.     fall back to that, access it from PowerPC asm. Overhead = 1.3 祍ec.
  29.   * We can also try Microseconds() which is emulated : Overhead = 36 祍ec.
  30. On PowerPC machines, we avoid the following:
  31.   * OpenTransport is available on all PCI and some NuBus PowerMacs, but it
  32.     uses UpTime() if available and falls back to Microseconds() otherwise.
  33.   * InputSprocket is available on many PowerMacs, but again it uses
  34.     UpTime() if available and falls back to Microseconds() otherwise.
  35. Another PowerPC note: certain configurations, especially 3rd party upgrade
  36. cards, may return inaccurate timings for the CPU or memory bus -- causing
  37. skew in various system routines (up to 20% drift!). The VIA chip is very
  38. accurate, and it's the basis for the Time Manager and Microseconds().
  39. Unfortunately, it's also very slow because the MacOS has to (a) switch to
  40. 68K and (b) poll for a VIA event.
  41. We compensate for the drift by calibrating a floating point scale factor
  42. between our fast method and the accurate timer at startup, then convert
  43. each sample quickly on the fly. I'd rather not have the initialization 
  44. overhead -- but it's simply necessary for accurate timing. You can drop
  45. it down to 30 ticks if you prefer, but that's as low as I'd recommend.
  46. Under MacOS 9, "new world" Macs (iMacs, B+W G3s and G+W G4s) have a native
  47. Time Manager implementation: UpTime(), Microseconds(), and TickCount() are
  48. all based on the same underlying counter. This makes it silly to calibrate
  49. UpTime() against TickCount(). We now check for this feature using Gestalt(),
  50. and skip the whole calibration step if possible.
  51. */
  52. /* **** **** **** **** **** **** **** **** **** **** **** **** **** **** **** */
  53. /* **** **** **** **** **** **** **** **** **** **** **** **** **** **** **** */
  54. #define RTCToNano(w) ((double) (w).hi * 1000000000.0 + (double) (w).lo)
  55. #define WideTo64bit(w) (*(UInt64 *) &(w))
  56. /* LMGetTicks() is not in Carbon and TickCount() has a fair bit of overhead,
  57.    so for speed we always read lowmem directly. This is a MacOS X no-no, but 
  58.    it always work on those systems that don't have a native Time Manager (ie,
  59.    anything before MacOS 9) -- regardless whether we are in Carbon or not! */
  60. #define MyLMGetTicks() (*(volatile UInt32 *) 0x16A)
  61. #if GENERATINGPOWERPC
  62. static asm UnsignedWide PollRTC(void);
  63. static asm UnsignedWide PollTBR(void);
  64. static Ptr FindFunctionInSharedLib(StringPtr libName, StringPtr funcName);
  65. static Boolean gInited = false;
  66. static Boolean gNative = false;
  67. static Boolean gUseRTC = false;
  68. static Boolean gUseTBR = false;
  69. static double gScaleUSec = 1.0 / 1000.0;    /* 1 / ( nsec / usec) */
  70. static double gScaleMSec = 1.0 / 1000000.0; /* 1 / ( nsec / msec) */
  71. /* Functions loaded from DriverServicesLib */
  72. typedef AbsoluteTime  (*UpTimeProcPtr)(void);
  73. typedef Nanoseconds  (*A2NSProcPtr)(AbsoluteTime);
  74. static UpTimeProcPtr  gUpTime = NULL;
  75. static A2NSProcPtr  gA2NS = NULL;
  76. #endif /* GENERATINGPOWERPC */
  77. /* **** **** **** **** **** **** **** **** **** **** **** **** **** **** **** */
  78. /* **** **** **** **** **** **** **** **** **** **** **** **** **** **** **** */
  79. void FastInitialize() {
  80. SInt32 result;
  81. if (!gInited) {
  82. #if GENERATINGPOWERPC
  83. /* Initialize the feature flags */
  84. gNative = gUseRTC = gUseTBR = false;
  85. /* We use CFM to find and load needed symbols from shared libraries, so
  86.    the application doesn't have to weak-link them, for convenience.   */
  87. gUpTime = (UpTimeProcPtr) FindFunctionInSharedLib(
  88. "pDriverServicesLib", "pUpTime");
  89. if (gUpTime) gA2NS = (A2NSProcPtr) FindFunctionInSharedLib(
  90. "pDriverServicesLib", "pAbsoluteToNanoseconds");
  91. if (!gA2NS) gUpTime = nil; /* Pedantic but necessary */
  92. if (gUpTime) {
  93. /* If we loaded UpTime(), then we need to know if the system has
  94.    a native implementation of the Time Manager. If so, then it's
  95.    pointless to calculate a scale factor against the missing VIA */
  96. /* gestaltNativeTimeMgr = 4 in some future version of the headers */
  97. if (!Gestalt(gestaltTimeMgrVersion, &result) &&
  98. (result > gestaltExtendedTimeMgr)) 
  99. gNative = true;
  100. }
  101.   else {
  102. /* If no DriverServicesLib, use Gestalt() to get the processor type. 
  103.    Only NuBus PowerMacs with old System Software won't have DSL, so
  104.    we know it should either be a 601 or 603. */
  105. /* Use the processor gestalt to determine which register to use */
  106.   if (!Gestalt(gestaltNativeCPUtype, &result)) {
  107. if (result == gestaltCPU601) gUseRTC = true;
  108.   else if (result > gestaltCPU601) gUseTBR = true;
  109. }
  110. }
  111. /* Now calculate a scale factor to keep us accurate. */
  112. if ((gUpTime && !gNative) || gUseRTC || gUseTBR) {
  113. UInt64 tick, usec1, usec2;
  114. UnsignedWide wide;
  115. /* Wait for the beginning of the very next tick */
  116. for(tick = MyLMGetTicks() + 1; tick > MyLMGetTicks(); );
  117. /* Poll the selected timer and prepare it (since we have time) */
  118. wide = (gUpTime) ? (*gA2NS)((*gUpTime)()) : 
  119. ((gUseRTC) ? PollRTC() : PollTBR());
  120. usec1 = (gUseRTC) ? RTCToNano(wide) : WideTo64bit(wide);
  121. /* Wait for the exact 60th tick to roll over */
  122. while(tick + 60 > MyLMGetTicks());
  123. /* Poll the selected timer again and prepare it  */
  124. wide = (gUpTime) ? (*gA2NS)((*gUpTime)()) : 
  125. ((gUseRTC) ? PollRTC() : PollTBR());
  126. usec2 = (gUseRTC) ? RTCToNano(wide) : WideTo64bit(wide);
  127. /* Calculate a scale value that will give microseconds per second.
  128.    Remember, there are actually 60.15 ticks in a second, not 60.  */
  129. gScaleUSec = (60.0 * 1000000.0) / ((usec2 - usec1) * 60.15);
  130. gScaleMSec = gScaleUSec / 1000.0;
  131. }
  132. #endif /* GENERATINGPOWERPC */
  133. /* We've initialized our globals */
  134. gInited = true;
  135. }
  136. }
  137. /* **** **** **** **** **** **** **** **** **** **** **** **** **** **** **** */
  138. /* **** **** **** **** **** **** **** **** **** **** **** **** **** **** **** */
  139. UInt64 FastMicroseconds() {
  140. UnsignedWide wide;
  141. UInt64 usec;
  142. #if GENERATINGPOWERPC
  143. /* Initialize globals the first time we are called */
  144. if (!gInited) FastInitialize();
  145. if (gNative) {
  146. /* Use DriverServices if it's available -- it's fast and compatible */
  147. wide = (*gA2NS)((*gUpTime)());
  148. usec = (double) WideTo64bit(wide) * gScaleUSec + 0.5;
  149. }
  150.   else if (gUpTime) {
  151. /* Use DriverServices if it's available -- it's fast and compatible */
  152. wide = (*gA2NS)((*gUpTime)());
  153. usec = (double) WideTo64bit(wide) * gScaleUSec + 0.5;
  154. }
  155.   else if (gUseTBR) {
  156. /* On a recent PowerPC, we poll the TBR directly */
  157. wide = PollTBR();
  158. usec = (double) WideTo64bit(wide) * gScaleUSec + 0.5;
  159. }
  160.   else if (gUseRTC) {
  161. /* On a 601, we can poll the RTC instead */
  162. wide = PollRTC();
  163. usec = (double) RTCToNano(wide) * gScaleUSec + 0.5;
  164. }
  165.   else 
  166. #endif /* GENERATINGPOWERPC */
  167. {
  168. /* If all else fails, suffer the mixed mode overhead */
  169. Microseconds(&wide);
  170. usec = WideTo64bit(wide);
  171. }
  172. return(usec);
  173. }
  174. /* **** **** **** **** **** **** **** **** **** **** **** **** **** **** **** */
  175. /* **** **** **** **** **** **** **** **** **** **** **** **** **** **** **** */
  176. UInt64 FastMilliseconds() {
  177. UnsignedWide wide;
  178. UInt64 msec;
  179. #if GENERATINGPOWERPC
  180. /* Initialize globals the first time we are called */
  181. if (!gInited) FastInitialize();
  182. if (gNative) {
  183. /* Use DriverServices if it's available -- it's fast and compatible */
  184. wide = (*gA2NS)((*gUpTime)());
  185. msec = (double) WideTo64bit(wide) * gScaleMSec + 0.5;
  186. }
  187.   else if (gUpTime) {
  188. /* Use DriverServices if it's available -- it's fast and compatible */
  189. wide = (*gA2NS)((*gUpTime)());
  190. msec = (double) WideTo64bit(wide) * gScaleMSec + 0.5;
  191. }
  192.   else if (gUseTBR) {
  193. /* On a recent PowerPC, we poll the TBR directly */
  194. wide = PollTBR();
  195. msec = (double) WideTo64bit(wide) * gScaleMSec + 0.5;
  196. }
  197.   else if (gUseRTC) {
  198. /* On a 601, we can poll the RTC instead */
  199. wide = PollRTC();
  200. msec = (double) RTCToNano(wide) * gScaleMSec + 0.5;
  201. }
  202.   else 
  203. #endif /* GENERATINGPOWERPC */
  204. {
  205. /* If all else fails, suffer the mixed mode overhead */
  206. Microseconds(&wide);
  207. msec = ((double) WideTo64bit(wide) + 500.0) / 1000.0;
  208. }
  209. return(msec);
  210. }
  211. /* **** **** **** **** **** **** **** **** **** **** **** **** **** **** **** */
  212. /* **** **** **** **** **** **** **** **** **** **** **** **** **** **** **** */
  213. StringPtr FastMethod() {
  214. StringPtr method = "p<Unknown>";
  215. #if GENERATINGPOWERPC
  216. /* Initialize globals the first time we are called */
  217. if (!gInited) FastInitialize();
  218. if (gNative) {
  219. /* The Time Manager and UpTime() are entirely native on this machine */
  220. method = "pNative UpTime()";
  221. }
  222.   else if (gUpTime) {
  223. /* Use DriverServices if it's available -- it's fast and compatible */
  224. method = "pUpTime()";
  225. }
  226.   else if (gUseTBR) {
  227. /* On a recent PowerPC, we poll the TBR directly */
  228. method = "pPowerPC TBR";
  229. }
  230.   else if (gUseRTC) {
  231. /* On a 601, we can poll the RTC instead */
  232. method = "pPowerPC RTC";
  233. }
  234.   else 
  235. #endif /* GENERATINGPOWERPC */
  236. {
  237. /* If all else fails, suffer the mixed mode overhead */
  238. method = "pMicroseconds()";
  239. }
  240. return(method);
  241. }
  242. /* **** **** **** **** **** **** **** **** **** **** **** **** **** **** **** */
  243. /* **** **** **** **** **** **** **** **** **** **** **** **** **** **** **** */
  244. #pragma mark -
  245. #if GENERATINGPOWERPC
  246. asm static UnsignedWide PollRTC_() {
  247. entry PollRTC /* Avoid CodeWarrior glue */
  248. machine 601
  249. @AGAIN:
  250. mfrtcu r4 /* RTCU = SPR 4 */
  251. mfrtcl r5 /* RTCL = SPR 5 */
  252. mfrtcu r6
  253. cmpw r4,r6
  254. bne @AGAIN
  255. stw r4,0(r3)
  256. stw r5,4(r3)
  257. blr
  258. }
  259. /* **** **** **** **** **** **** **** **** **** **** **** **** **** **** **** */
  260. /* **** **** **** **** **** **** **** **** **** **** **** **** **** **** **** */
  261. asm static UnsignedWide PollTBR_() {
  262. entry PollTBR /* Avoid CodeWarrior glue */
  263. machine 604
  264. @AGAIN:
  265. mftbu r4 /* TBRU = SPR 268 */
  266. mftb r5 /* TBRL = SPR 269 */
  267. mftbu r6
  268. cmpw r4,r6
  269. bne @AGAIN
  270. stw r4,0(r3)
  271. stw r5,4(r3)
  272. blr
  273. }
  274. /* **** **** **** **** **** **** **** **** **** **** **** **** **** **** **** */
  275. /* **** **** **** **** **** **** **** **** **** **** **** **** **** **** **** */
  276. static Ptr FindFunctionInSharedLib(StringPtr libName, StringPtr funcName) {
  277. OSErr error = noErr;
  278. Str255 errorStr;
  279. Ptr func = NULL;
  280. Ptr entry = NULL;
  281. CFragSymbolClass symClass;
  282. CFragConnectionID connID;
  283. /* Find CFM containers for the current archecture -- CFM-PPC or CFM-68K */
  284. if (/* error = */ GetSharedLibrary(libName, kCompiledCFragArch,
  285. kLoadCFrag, &connID, &entry, errorStr)) return(NULL);
  286. if (/* error = */ FindSymbol(connID, funcName, &func, &symClass))
  287. return(NULL);
  288. return(func);
  289. }
  290. #endif /* GENERATINGPOWERPC */