mmxcpuid.c
上传用户:zhongxx05
上传日期:2007-06-06
资源大小:33641k
文件大小:5k
源码类别:

Symbian

开发平台:

C/C++

  1. /* ***** BEGIN LICENSE BLOCK ***** 
  2.  * Version: RCSL 1.0/RPSL 1.0 
  3.  *  
  4.  * Portions Copyright (c) 1995-2002 RealNetworks, Inc. All Rights Reserved. 
  5.  *      
  6.  * The contents of this file, and the files included with this file, are 
  7.  * subject to the current version of the RealNetworks Public Source License 
  8.  * Version 1.0 (the "RPSL") available at 
  9.  * http://www.helixcommunity.org/content/rpsl unless you have licensed 
  10.  * the file under the RealNetworks Community Source License Version 1.0 
  11.  * (the "RCSL") available at http://www.helixcommunity.org/content/rcsl, 
  12.  * in which case the RCSL will apply. You may also obtain the license terms 
  13.  * directly from RealNetworks.  You may not use this file except in 
  14.  * compliance with the RPSL or, if you have a valid RCSL with RealNetworks 
  15.  * applicable to this file, the RCSL.  Please see the applicable RPSL or 
  16.  * RCSL for the rights, obligations and limitations governing use of the 
  17.  * contents of the file.  
  18.  *  
  19.  * This file is part of the Helix DNA Technology. RealNetworks is the 
  20.  * developer of the Original Code and owns the copyrights in the portions 
  21.  * it created. 
  22.  *  
  23.  * This file, and the files included with this file, is distributed and made 
  24.  * available on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER 
  25.  * EXPRESS OR IMPLIED, AND REALNETWORKS HEREBY DISCLAIMS ALL SUCH WARRANTIES, 
  26.  * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY, FITNESS 
  27.  * FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT. 
  28.  * 
  29.  * Technology Compatibility Kit Test Suite(s) Location: 
  30.  *    http://www.helixcommunity.org/content/tck 
  31.  * 
  32.  * Contributor(s): 
  33.  *  
  34.  * ***** END LICENSE BLOCK ***** */ 
  35. #include "mmxcpuid.h"
  36. /* GLOBALS */
  37. long g_mmx_checked = CPUID_NOTINIT;
  38. long g_mmx_detected = CPUID_MMX_OFF;
  39. /* externs */
  40. long VvGetPrefPrivateProfileInt(char *, long);
  41. #ifdef COMPILE_MMX
  42. #pragma message("Compiling MMX code, CPU detection enabled with cpuid_init()")
  43. #endif
  44. /*----------------------------------------------------------------------------*/ 
  45. long __stdcall mmxsupport()
  46. {
  47. int mmx_supported = 0;
  48. int *pmmx_supported = &mmx_supported;
  49. //NOTE: IMPORTANT
  50. //MSVC 5.0 Compiler with optimization "maximize speed" will obviously 
  51. // try to minimize registers saved for _asm blocks
  52. // but it does not have any way to tell that there is a CPUID instruction and that this instruction 
  53. // will affect eax, ebx, ecx, edx. 
  54. // be sure to use all the registers in the _asm block so that the compiler will save them before entering
  55. _asm {
  56. pushfd //Save Eflag to stack
  57. pop eax //Get Eflag from stack into eax
  58. mov ecx, eax //Make another copy of Eflag in ecx
  59. xor eax, 0x200000 //Toggle ID bit in Eflag [i.e. bit(21)] 
  60. push eax //Save modified Eflag back to stack
  61. popfd //Restore modified value back to Eflag reg 
  62. pushfd //Save Eflag to stack
  63. pop eax //Get Eflag from stack
  64. xor eax, ecx //Compare the new Eflag with the original Eflag
  65. jz NOT_SUPPORTED //If the same, CPUID instruction is not supported,
  66. //skip following instructions and jump to
  67. //NOT_SUPPORTED label
  68. xor eax, eax //Set eax to zero
  69. _asm _emit 0x0f //CPUID instruction  (two bytes opcode)
  70. _asm _emit 0xa2 
  71. cmp eax, 1 //make sure eax return non-zero value
  72. jl NOT_SUPPORTED //If eax is zero, mmx not supported
  73. xor eax, eax //set eax to zero
  74. inc eax //Now increment eax to 1.  This instruction is 
  75. //faster than the instruction "mov eax, 1"
  76. _asm _emit 0x0f //CPUID instruction
  77. _asm _emit 0xa2 
  78. and edx, 0x00800000 //mask out all bits but mmx bit(24)
  79. cmp edx, 0 // 0 = mmx not supported
  80. jz NOT_SUPPORTED // non-zero = Yes, mmx IS supported
  81. mov ebx, pmmx_supported
  82. mov [ebx], 1 //set return value to 1
  83. NOT_SUPPORTED:
  84. nop
  85. //mov eax, mmx_supported //move return value to eax
  86. }
  87. return mmx_supported;
  88. }
  89. /*----------------------------------------------------------------------------*/ 
  90. void cpuid_init() 
  91. {
  92. long mmx = 0;
  93. long temp;//only word needed
  94. //mask all unnecessary floating point exceptions 
  95. //on fresh reboot, denorm is not masked at this point
  96. __asm {
  97. push eax
  98. fclex ;//clear exceptions
  99. fstcw temp;
  100. mov eax, temp
  101. or eax, 0x000f
  102. mov temp, eax
  103. fldcw temp;
  104. pop eax
  105. }
  106. //VIVOGEN_BITRATE_BASE-style in vdefine.h would not work for .ini files
  107. #if defined(_WIN32) || defined(_LINUX)
  108. g_mmx_detected = 1;
  109. #else
  110. g_mmx_detected = 0;
  111. #endif
  112. if(g_mmx_detected != 0) {//enabled, find out if we are on a MMX processor
  113. mmx = mmxsupport();
  114. if(mmx == 0) {//turned off because not on a MMX processor
  115. g_mmx_detected = CPUID_MMX_OFF;
  116. }
  117. } else {//turned off in ini file or registry
  118. g_mmx_detected = CPUID_MMX_OFF;
  119. }
  120. g_mmx_checked = CPUID_VALID;
  121. }