Assembly CPU frequency measuring algorithm
Asked Answered
M

10

18

What are the common algorithms being used to measure the processor frequency?

Morse answered 15/9, 2008 at 17:50 Comment(0)
C
21

Intel CPUs after Core Duo support two Model-Specific registers called IA32_MPERF and IA32_APERF.
MPERF counts at the maximum frequency the CPU supports, while APERF counts at the actual current frequency.

The actual frequency is given by:

freq = max_frequency * APERF / MPERF

You can read them with this flow

; read MPERF
mov ecx, 0xe7
rdmsr
mov mperf_var_lo, eax
mov mperf_var_hi, edx

; read APERF
mov ecx, 0xe8
rdmsr
mov aperf_var_lo, eax
mov aperf_var_hi, edx

but note that rdmsr is a privileged instruction and can run only in ring 0.

I don't know if the OS provides an interface to read these, though their main usage is for power management, so it might not provide such an interface.

Caseinogen answered 15/9, 2008 at 18:20 Comment(4)
I used this site to convert LaTeX code to GIF: en.wikibooks.org/wiki/LaTeX/Mathematics. LaTeX syntax for mathematics is described here: en.wikibooks.org/wiki/LaTeX/Mathematics.Antidote
@NathanFellman Nathan, but how do you get the maximum frequency?Conakry
@NathanFellman I wanna make two points about my comment/question on the maximum frequency. One, from a kernel driver you can't just use QueryPerformanceFrequency to get this value (won't allow you to use Windows.h header, you need to use a special kernel header instead with no access to this method). You probably could pass the MSR values to a user mode app and calculate it like this but...point two, even if you could use it, it wouldn't give you an affinate thread's frequency but the frequency of the entire CPU instead, which sucks if you want to know the frequency of just one core.Conakry
@Alexandru, sorry it's taken me so long to get back to you on this. For your 1st question about max frequency, you can use CPUID.0x16 to get the max non-turbo frequency. I'm not sure if this leaf existed when you asked your question, but it does now. Regarding your 2nd question, these MSRs are logical-processor-scoped, so each individual thread has its own set of counters, so in fact this method will in fact give you the average frequency of just one thread.Caseinogen
W
7

I'm gonna date myself with various details in this answer, but what the heck...

I had to tackle this problem years ago on Windows-based PCs, so I was dealing with Intel x86 series processors like 486, Pentium and so on. The standard algorithm in that situation was to do a long series of DIVide instructions, because those are typically the most CPU-bound single instructions in the Intel set. So memory prefetch and other architectural issues do not materially affect the instruction execution time -- the prefetch queue is always full and the instruction itself does not touch any other memory.

You would time it using the highest resolution clock you could get access to in the environment you are running in. (In my case I was running near boot time on a PC compatible, so I was directly programming the timer chips on the motherboard. Not recommended in a real OS, usually there's some appropriate API to call these days).

The main problem you have to deal with is different CPU types. At that time there was Intel, AMD and some smaller vendors like Cyrix making x86 processors. Each model had its own performance characteristics vis-a-vis that DIV instruction. My assembly timing function would just return a number of clock cycles taken by a certain fixed number of DIV instructions done in a tight loop.

So what I did was to gather some timings (raw return values from that function) from actual PCs running each processor model I wanted to time, and record those in a spreadsheet against the known processor speed and processor type. I actually had a command-line tool that was just a thin shell around my timing function, and I would take a disk into computer stores and get the timings off of display models! (I worked for a very small company at the time).

Using those raw timings, I could plot a theoretical graph of what timings I should get for any known speed of that particular CPU.

Here was the trick: I always hated when you would run a utility and it would announce that your CPU was 99.8 Mhz or whatever. Clearly it was 100 Mhz and there was just a small round-off error in the measurement. In my spreadsheet I recorded the actual speeds that were sold by each processor vendor. Then I would use the plot of actual timings to estimate projected timings for any known speed. But I would build a table of points along the line where the timings should round to the next speed.

In other words, if 100 ticks to do all that repeating dividing meant 500 Mhz, and 200 ticks meant 250 Mhz, then I would build a table that said that anything below 150 was 500 Mhz, and anything above that was 250 Mhz. (Assuming those were the only two speeds available from that chip vendor). It was nice because even if some odd piece of software on the PC was throwing off my timings, the end result would often still be dead on.

Of course now, in these days of overclocking, dynamic clock speeds for power management, and other such trickery, such a scheme would be much less practical. At the very least you'd need to do something to make sure the CPU was in its highest dynamically chosen speed first before running your timing function.

OK, I'll go back to shooing kids off my lawn now.

Wandie answered 23/9, 2008 at 0:16 Comment(2)
I'd say that a speed of 99.8 could be accurate, they were aiming for 100Mhz but missed. You don't expect them to cull system chips that are a bit off timewise like you would wristwatches.Jolyn
I agree. Not only you have the aim theory, but also you have factors that you have to expect a level of fluctuations in a computer. Motherboards actually take in a fluctuating amount of voltages. A "100 MHz" CPU also could actually be a 25 MHz CPU with a 4x multiplier or a 33 MHz with a 3x multiplierMandler
S
4

One way on x86 Intel CPU's since Pentium would be to use two samplings of the RDTSC instruction with a delay loop of known wall time, eg:

#include <stdio.h>
#include <stdint.h>
#include <unistd.h>

uint64_t rdtsc(void) {
    uint64_t result;
    __asm__ __volatile__ ("rdtsc" : "=A" (result));
    return result;
}

int main(void) {
    uint64_t ts0, ts1;    
    ts0 = rdtsc();
    sleep(1);
    ts1 = rdtsc();    
    printf("clock frequency = %llu\n", ts1 - ts0);
    return 0;
}

(on 32-bit platforms with GCC)

RDTSC is available in ring 3 if the TSC flag in CR4 is set, which is common but not guaranteed. One shortcoming of this method is that it is vulnerable to frequency scaling changes affecting the result if they happen inside the delay. To mitigate that you could execute code that keeps the CPU busy and constantly poll the system time to see if your delay period has expired, to keep the CPU in the highest frequency state available.

Shied answered 26/6, 2009 at 9:6 Comment(1)
RDTSC counts "reference cycles" on modern CPUs, not core clock cycles. So this will determine the "rated" / max-sustained clock frequency on most modern CPUs, not the current clocks. This may be what you actually want, except on overclocked systems that run at a different speed permanently.Antecedents
R
2

I use the following (pseudo)algorithm:

basetime=time();    /* time returns seconds */

while (time()==basetime);
stclk=rdtsc();    /* rdtsc is an assembly instruction */

basetime=time();
while (time()==basetime
endclk=rdtsc();

nclks=encdclk-stclk;

At this point you might assume that you've determined the clock frequency but even though it appears correct it can be improved.

All PCs contain a PIT (Programmable Interval Timer) device which contains counters which are (used to be) used for serial ports and the system clock. It was fed with a frequency of 1193182 Hz. The system clock counter was set to the highest countdown value (65536) resulting in a system clock tick frequency of 1193182/65536 => 18.2065 Hz or once every 54.925 milliseconds.

The number of ticks necessary for the clock to increment to the next second will therefore depend. Usually 18 ticks are required and sometimes 19. This can be handled by performing the algorithm (above) twice and storing the results. The two results will either be equivalent to two 18 tick sequences or one 18 and one 19. Two 19s in a row won't occur. So by taking the smaller of the two results you will have an 18 tick second. Adjust this result by multiplying with 18.2065 and dividing by 18.0 or, using integer arithmetic, multiply by 182065, add 90000 and divide by 180000. 90000 is one half of 180000 and is there for rounding. If you choose the calculation with integer route make sure you are using 64-bit multiplication and division.

You will now have a CPU clock speed x in Hz which can be converted to kHz ((x+500)/1000) or MHz ((x+5000000)/1000000). The 500 and 500000 are one half of 1000 and 1000000 respectively and are there for rounding. To calculate MHz do not go via the kHz value because rounding issues may arise. Use the Hz value and the second algorithm.

Restrainer answered 5/8, 2011 at 9:7 Comment(0)
P
2

One option is to sense the CPU frequency, by running code with known instructions per loop

This functionality is contained in 7zip, since about v9.20 I think.

> 7z b
7-Zip 9.38 beta  Copyright (c) 1999-2014 Igor Pavlov  2015-01-03

CPU Freq:  4266  4000  4266  4000  2723  4129  3261  3644  3362

The final number is meant to be correct (and on my PC and many others, I have found it to be quite correct - the test runs very quick so turbo may not kick in, and servers set in Balanced/Power Save modes most likely give readings of around 1ghz)

The source code is at GitHub (Official source is a download from 7-zip.org)

With the most significant portion being:

#define YY1 sum += val; sum ^= val;
#define YY3 YY1 YY1 YY1 YY1
#define YY5 YY3 YY3 YY3 YY3
#define YY7 YY5 YY5 YY5 YY5
static const UInt32 kNumFreqCommands = 128;

EXTERN_C_BEGIN

static UInt32 CountCpuFreq(UInt32 sum, UInt32 num, UInt32 val)
{
  for (UInt32 i = 0; i < num; i++)
  {
    YY7
  }
  return sum;
}

EXTERN_C_END
Presidentelect answered 16/10, 2015 at 0:20 Comment(0)
M
2

On Intel CPUs, a common method to get the current (average) CPU frequency is to calculate it from a few CPU counters:

CPU_freq = tsc_freq * (aperf_t1 - aperf_t0) / (mperf_t1 - mperf_t0)

The TSC (Time Stamp Counter) can be read from userspace with dedicated x86 instructions, but its frequency has to be determined by calibration against a clock. The best approach is to get the TSC frequency from the kernel (which already has done the calibration).

The aperf and mperf counters are model specific registers MSRs that require root privileges for access. Again, there are dedicated x86 instructions for accessing the MSRs.

Since the mperf counter rate is directly proportional to the TSC rate and the aperf rate is directly proportional to the CPU frequency you get the CPU frequency with the above equation.

Of course, if the CPU frequency changes in your t0 - t1 time delta (e.g. due due frequency scaling) you get the average CPU frequency with this method.

I wrote a small utility cpufreq which can be used to test this method.

See also:

Mize answered 25/4, 2020 at 16:44 Comment(4)
Thanks this was helpeful. Also see patchwork.kernel.org/project/linux-pm/patch/…, where they note that for instantaneous frequency you can read MSR_PERF_STATUS but this requires the OS to enable frequency control and actually manipulate the valueKrahmer
Also see intel manual xem.github.io/minix86/manual/intel-x86-and-64-manual-vol3/… By default, the IA32_MPERF counter counts during forced idle periods as if the logical processor was active. The IA32_APERF counter does not count during forced idle state. This counting convention allows the OS to compute the average effective frequency of the Logical Processor between the last MWAIT exit and the next MWAIT entry (OS visible C0) by ΔACNT/ΔMCNT * TSC FrequencyKrahmer
And finally, as for why some answers state nominal frequency (max is misleading, since you want to use nominal i.e. non-trubo max) instead of TSC frequency, per intel That rate may be set by the maximum core-clock to bus-clock ratio of the processor or may be set by the maximum resolved frequency at which the processor is booted. The maximum resolved frequency may differ from the processor base frequency, see Section 18.7.2 for more detail. On certain processors, the TSC frequency may not be the same as the frequency brand string.Krahmer
@Krahmer Thanks for the additional references - if you found the answer helpful you can also up-vote it ... :)Mize
S
1

That was the intention of things like BogoMIPS, but CPUs are a lot more complicated nowadays. Superscalar CPUs can issue multiple instructions per clock, making any measurement based on counting clock cycles to execute a block of instructions highly inaccurate.

CPU frequencies are also variable based on offered load and/or temperature. The fact that the CPU is currently running at 800 MHz does not mean it will always be running at 800 MHz, it might throttle up or down as needed.

If you really need to know the clock frequency, it should be passed in as a parameter. An EEPROM on the board would supply the base frequency, and if the clock can vary you'd need to be able to read the CPUs power state registers (or make an OS call) to find out the frequency at that instant.

With all that said, there may be other ways to accomplish what you're trying to do. For example if you want to make high-precision measurements of how long a particular codepath takes, the CPU likely has performance counters running at a fixed frequency which are a better measure of wall-clock time than reading a tick count register.

Starryeyed answered 15/9, 2008 at 17:59 Comment(0)
C
1

"lmbench" provides a cpu frequency algorithm portable for different architecture.

It runs some different loops and the processor's clock speed is the greatest common divisor of the execution frequencies of the various loops.

this method should always work when we are able to get loops with cycle counts that are relatively prime.

http://www.bitmover.com/lmbench/

Cock answered 27/3, 2009 at 9:50 Comment(0)
T
0

I'm not sure why you need assembly for this. If you're on a machine that has the /proc filesystem, then running:

> cat /proc/cpuinfo

might give you what you need.

Taddeusz answered 15/9, 2008 at 17:57 Comment(0)
W
0

A quick google on AMD and Intel shows that CPUID should give you access to the CPU`s max frequency.

Waller answered 7/1, 2009 at 2:4 Comment(1)
I think it will only identify the processor model.Antidote

© 2022 - 2024 — McMap. All rights reserved.