Fixed the bus and core clock reading on AMD family 10h model Ah CPUs. The new "Core Performance Boost" feature of these CPUs resulted in very low accuracy of the bus speed (and as a consequence also an inaccurate TSC multiplier). This fixed Issue 205.
authormoel.mich
Sun, 08 May 2011 22:10:13 +0000
changeset 2796bce967ba1b5
parent 278 803f98bf1418
child 280 2bc456906ad6
Fixed the bus and core clock reading on AMD family 10h model Ah CPUs. The new "Core Performance Boost" feature of these CPUs resulted in very low accuracy of the bus speed (and as a consequence also an inaccurate TSC multiplier). This fixed Issue 205.
Hardware/CPU/AMD10CPU.cs
Hardware/CPU/GenericCPU.cs
Hardware/Ring0.cs
Properties/AssemblyVersion.cs
     1.1 --- a/Hardware/CPU/AMD10CPU.cs	Tue May 03 18:20:06 2011 +0000
     1.2 +++ b/Hardware/CPU/AMD10CPU.cs	Sun May 08 22:10:13 2011 +0000
     1.3 @@ -54,6 +54,7 @@
     1.4        
     1.5      private const uint PERF_CTL_0 = 0xC0010000;
     1.6      private const uint PERF_CTR_0 = 0xC0010004;
     1.7 +    private const uint HWCR = 0xC0010015;
     1.8      private const uint P_STATE_0 = 0xC0010064;
     1.9      private const uint COFVID_STATUS = 0xC0010071;
    1.10  
    1.11 @@ -69,7 +70,8 @@
    1.12  
    1.13      private readonly FileStream temperatureStream;
    1.14  
    1.15 -    private double timeStampCounterMultiplier;
    1.16 +    private readonly double timeStampCounterMultiplier;
    1.17 +    private readonly bool corePerformanceBoostSupport;
    1.18  
    1.19      public AMD10CPU(int processorIndex, CPUID[][] cpuid, ISettings settings)
    1.20        : base(processorIndex, cpuid, settings) 
    1.21 @@ -104,9 +106,17 @@
    1.22            ActivateSensor(coreClocks[i]);
    1.23        }
    1.24  
    1.25 +      corePerformanceBoostSupport = (cpuid[0][0].ExtData[7, 3] & (1 << 9)) > 0;
    1.26 +
    1.27        // set affinity to the first thread for all frequency estimations     
    1.28        ulong mask = ThreadAffinity.Set(1UL << cpuid[0][0].Thread);
    1.29  
    1.30 +      // disable core performance boost  
    1.31 +      uint hwcrEax, hwcrEdx;
    1.32 +      Ring0.Rdmsr(HWCR, out hwcrEax, out hwcrEdx);
    1.33 +      if (corePerformanceBoostSupport) 
    1.34 +        Ring0.Wrmsr(HWCR, hwcrEax | (1 << 25), hwcrEdx);
    1.35 +
    1.36        uint ctlEax, ctlEdx;
    1.37        Ring0.Rdmsr(PERF_CTL_0, out ctlEax, out ctlEdx);
    1.38        uint ctrEax, ctrEdx;
    1.39 @@ -118,6 +128,10 @@
    1.40        Ring0.Wrmsr(PERF_CTL_0, ctlEax, ctlEdx);
    1.41        Ring0.Wrmsr(PERF_CTR_0, ctrEax, ctrEdx);
    1.42  
    1.43 +      // restore core performance boost
    1.44 +      if (corePerformanceBoostSupport)     
    1.45 +        Ring0.Wrmsr(HWCR, hwcrEax, hwcrEdx);
    1.46 +
    1.47        // restore the thread affinity.
    1.48        ThreadAffinity.Set(mask);
    1.49  
    1.50 @@ -178,10 +192,11 @@
    1.51        long timeEnd = timeBegin + ticks;
    1.52        while (Stopwatch.GetTimestamp() < timeBegin) { }
    1.53        Ring0.Rdmsr(PERF_CTR_0, out lsbBegin, out msbBegin);
    1.54 +
    1.55        while (Stopwatch.GetTimestamp() < timeEnd) { }
    1.56        Ring0.Rdmsr(PERF_CTR_0, out lsbEnd, out msbEnd);
    1.57 +      Ring0.Rdmsr(COFVID_STATUS, out eax, out edx);
    1.58  
    1.59 -      Ring0.Rdmsr(COFVID_STATUS, out eax, out edx);
    1.60        double coreMultiplier;
    1.61        if (family == 0x14) {               
    1.62          uint divisorIdMSD = (eax >> 4) & 0x1F;
    1.63 @@ -206,11 +221,13 @@
    1.64          (timeEnd - timeBegin);
    1.65  
    1.66        double busFrequency = coreFrequency / coreMultiplier;
    1.67 +
    1.68        return 0.25 * Math.Round(4 * TimeStampCounterFrequency / busFrequency);
    1.69      }
    1.70  
    1.71      protected override uint[] GetMSRs() {
    1.72 -      return new uint[] { PERF_CTL_0, PERF_CTR_0, P_STATE_0, COFVID_STATUS };
    1.73 +      return new uint[] { PERF_CTL_0, PERF_CTR_0, HWCR, P_STATE_0, 
    1.74 +        COFVID_STATUS };
    1.75      }
    1.76  
    1.77      public override string GetReport() {
     2.1 --- a/Hardware/CPU/GenericCPU.cs	Tue May 03 18:20:06 2011 +0000
     2.2 +++ b/Hardware/CPU/GenericCPU.cs	Sun May 08 22:10:13 2011 +0000
     2.3 @@ -60,10 +60,12 @@
     2.4      private readonly bool hasTimeStampCounter;
     2.5      private readonly bool isInvariantTimeStampCounter;
     2.6      private readonly double estimatedTimeStampCounterFrequency;
     2.7 +    private readonly double estimatedTimeStampCounterFrequencyError;
     2.8  
     2.9      private ulong lastTimeStampCount;
    2.10      private long lastTime;
    2.11 -    private double timeStampCounterFrequency;    
    2.12 +    private double timeStampCounterFrequency;
    2.13 +    
    2.14  
    2.15      private readonly Vendor vendor;
    2.16  
    2.17 @@ -132,9 +134,10 @@
    2.18  
    2.19        if (hasTimeStampCounter) {
    2.20          ulong mask = ThreadAffinity.Set(1UL << cpuid[0][0].Thread);
    2.21 -        
    2.22 -        estimatedTimeStampCounterFrequency = 
    2.23 -          EstimateTimeStampCounterFrequency();  
    2.24 +
    2.25 +        EstimateTimeStampCounterFrequency(
    2.26 +          out estimatedTimeStampCounterFrequency, 
    2.27 +          out estimatedTimeStampCounterFrequencyError);  
    2.28          
    2.29          ThreadAffinity.Set(mask);
    2.30        } else {
    2.31 @@ -157,34 +160,55 @@
    2.32          processorIndex.ToString(CultureInfo.InvariantCulture));
    2.33      }
    2.34  
    2.35 -    private static double EstimateTimeStampCounterFrequency() {           
    2.36 +    private void EstimateTimeStampCounterFrequency(out double frequency, 
    2.37 +      out double error) 
    2.38 +  {     
    2.39 +      double f, e;
    2.40 +      
    2.41        // preload the function
    2.42 -      EstimateTimeStampCounterFrequency(0);
    2.43 -      EstimateTimeStampCounterFrequency(0);
    2.44 +      EstimateTimeStampCounterFrequency(0, out f, out e);
    2.45 +      EstimateTimeStampCounterFrequency(0, out f, out e);
    2.46  
    2.47 -      // estimate the frequency in MHz      
    2.48 -      List<double> estimatedFrequency = new List<double>(3);
    2.49 -      for (int i = 0; i < 3; i++)
    2.50 -        estimatedFrequency.Add(1e-6 * EstimateTimeStampCounterFrequency(0.025));
    2.51 -                 
    2.52 -      estimatedFrequency.Sort();
    2.53 -      return estimatedFrequency[1];
    2.54 +      // estimate the frequency
    2.55 +      error = double.MaxValue;
    2.56 +      frequency = 0;
    2.57 +      for (int i = 0; i < 5; i++) {
    2.58 +        EstimateTimeStampCounterFrequency(0.025, out f, out e);
    2.59 +        if (e < error) {
    2.60 +          error = e;
    2.61 +          frequency = f;
    2.62 +        }
    2.63 +
    2.64 +        if (error < 1e-4)
    2.65 +          break;
    2.66 +      }                
    2.67      }
    2.68  
    2.69 -    private static double EstimateTimeStampCounterFrequency(double timeWindow) {
    2.70 +    private void EstimateTimeStampCounterFrequency(double timeWindow, 
    2.71 +      out double frequency, out double error) 
    2.72 +    {
    2.73        long ticks = (long)(timeWindow * Stopwatch.Frequency);
    2.74        ulong countBegin, countEnd;
    2.75  
    2.76        long timeBegin = Stopwatch.GetTimestamp() +
    2.77          (long)Math.Ceiling(0.001 * ticks);
    2.78        long timeEnd = timeBegin + ticks;
    2.79 +
    2.80        while (Stopwatch.GetTimestamp() < timeBegin) { }
    2.81        countBegin = Opcode.Rdtsc();
    2.82 +      long afterBegin = Stopwatch.GetTimestamp();
    2.83 +
    2.84        while (Stopwatch.GetTimestamp() < timeEnd) { }
    2.85        countEnd = Opcode.Rdtsc();
    2.86 +      long afterEnd = Stopwatch.GetTimestamp();
    2.87  
    2.88 -      return (((double)(countEnd - countBegin)) * Stopwatch.Frequency) /
    2.89 -        (timeEnd - timeBegin);
    2.90 +      double delta = (timeEnd - timeBegin);
    2.91 +      frequency = 1e-6 * 
    2.92 +        (((double)(countEnd - countBegin)) * Stopwatch.Frequency) / delta;
    2.93 +
    2.94 +      double beginError = (afterBegin - timeBegin) / delta;
    2.95 +      double endError = (afterEnd - timeEnd) / delta;
    2.96 +      error = beginError + endError;
    2.97      }
    2.98  
    2.99  
   2.100 @@ -225,6 +249,13 @@
   2.101        r.AppendLine("Time Stamp Counter: " + (hasTimeStampCounter ? (
   2.102          isInvariantTimeStampCounter ? "Invariant" : "Not Invariant") : "None"));
   2.103        r.AppendLine(string.Format(CultureInfo.InvariantCulture,
   2.104 +        "Estimated Time Stamp Counter Frequency: {0} MHz",
   2.105 +        Math.Round(estimatedTimeStampCounterFrequency * 100) * 0.01));
   2.106 +      r.AppendLine(string.Format(CultureInfo.InvariantCulture,
   2.107 +        "Estimated Time Stamp Counter Frequency Error: {0} Mhz",
   2.108 +        Math.Round(estimatedTimeStampCounterFrequency *
   2.109 +        estimatedTimeStampCounterFrequencyError * 1e5) * 1e-5));
   2.110 +      r.AppendLine(string.Format(CultureInfo.InvariantCulture,
   2.111          "Time Stamp Counter Frequency: {0} MHz",
   2.112          Math.Round(timeStampCounterFrequency * 100) * 0.01));   
   2.113        r.AppendLine();
     3.1 --- a/Hardware/Ring0.cs	Tue May 03 18:20:06 2011 +0000
     3.2 +++ b/Hardware/Ring0.cs	Sun May 08 22:10:13 2011 +0000
     3.3 @@ -66,7 +66,9 @@
     3.4        IOCTL_OLS_READ_PCI_CONFIG = new IOControlCode(OLS_TYPE, 0x851, 
     3.5          IOControlCode.Access.Read),
     3.6        IOCTL_OLS_WRITE_PCI_CONFIG = new IOControlCode(OLS_TYPE, 0x852,
     3.7 -        IOControlCode.Access.Write);
     3.8 +        IOControlCode.Access.Write),
     3.9 +      IOCTL_OLS_READ_MEMORY = new IOControlCode(OLS_TYPE, 0x841,
    3.10 +        IOControlCode.Access.Read);
    3.11  
    3.12      private static bool ExtractDriver(string fileName) {
    3.13        string resourceName = "OpenHardwareMonitor.Hardware." +
    3.14 @@ -310,5 +312,26 @@
    3.15  
    3.16        return driver.DeviceIOControl(IOCTL_OLS_WRITE_PCI_CONFIG, input);
    3.17      }
    3.18 +
    3.19 +    [StructLayout(LayoutKind.Sequential, Pack = 1)]
    3.20 +    private struct ReadMemoryInput {
    3.21 +      public ulong address;
    3.22 +      public uint unitSize;
    3.23 +      public uint count;
    3.24 +    }
    3.25 +
    3.26 +    public static bool ReadMemory<T>(ulong address, ref T buffer) {
    3.27 +      if (driver == null) {
    3.28 +        return false;
    3.29 +      }
    3.30 +
    3.31 +      ReadMemoryInput input = new ReadMemoryInput();
    3.32 +      input.address = address;
    3.33 +      input.unitSize = 1;
    3.34 +      input.count = (uint)Marshal.SizeOf(buffer);
    3.35 +
    3.36 +      return driver.DeviceIOControl(IOCTL_OLS_READ_MEMORY, input,
    3.37 +        ref buffer);
    3.38 +    }
    3.39    }
    3.40  }
     4.1 --- a/Properties/AssemblyVersion.cs	Tue May 03 18:20:06 2011 +0000
     4.2 +++ b/Properties/AssemblyVersion.cs	Sun May 08 22:10:13 2011 +0000
     4.3 @@ -37,5 +37,5 @@
     4.4  
     4.5  using System.Reflection;
     4.6  
     4.7 -[assembly: AssemblyVersion("0.3.0.3")]
     4.8 -[assembly: AssemblyInformationalVersion("0.3.0.3 Alpha")]
     4.9 \ No newline at end of file
    4.10 +[assembly: AssemblyVersion("0.3.0.5")]
    4.11 +[assembly: AssemblyInformationalVersion("0.3.0.5 Alpha")]
    4.12 \ No newline at end of file