mmxcpuid.c
上传用户:dangjiwu
上传日期:2013-07-19
资源大小:42019k
文件大小:6k
源码类别:

Symbian

开发平台:

Visual C++

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