Skip to content

Latest commit

 

History

History
87 lines (67 loc) · 3.3 KB

README.md

File metadata and controls

87 lines (67 loc) · 3.3 KB

CPUmeter

Empirically determine "cpu load" by comparing current loop() update rate against the historical best case.

This is a little code snippet which might be useful elsewhere...

Usage

Include the library:

#include <CPUmeter.h>
CPUmeter meter;

Inside setup(), create your output Stream. For example, use the Serial Stream:

void setup() {
  ...
  Serial.begin(115200);
  ...
}

Each time loop() runs, update the meter:

void loop() {
  ...
  meter.loopUpdate();
  ...
}

When you want to learn about CPU loading, send a report to the Stream.

  ...
  // short summary
  meter.reportTo(Serial);

  // more detail
  meter.longReportTo(Serial);
  ...

Example reports

meter.reportTo(Serial) might report

CPU load about 22%

meter.longReportTo(Serial) might report

Best 24149 Worst 2059 current 18808 loops/sec
 Worst delay 159 ms
CPU load about 22%

Simple access to summaries

Some people may want to format their own summary, or route it to a bar graph, etc.. Here are some helper methods.

int meter.getPercentLoad() returns the most recent percent CPU load measurment.

int meter.getWorstDelay() returns the slowest loop repetition time found.

int meter.getDeadlinesMissed() returns the number of times the deadline was missed so far

Configuration

meter.setSampleInterval(int millisec) The default meaurement rate counts loops every 5 seconds. If you need a 2 second rate, you can set it with setSampleInterval(2000);

meter.resetStats() This resets the meter's collected statistics and starts accumulating all over again. It might be useful if you want to have a fresh start without resetting the sketch, or skip past a known startup delay.

meter.setLoopDeadline(int millisec) The CPUmeter can also see if loop() takes "too long" to repeat. This could cause music to stutter, for example. Setting a loop deadline starts checking for that excessive delay. Setting a loop deadline adds an extra line to the report. For example, after calling cpuMeter.setLoopDeadline(50) might add this line to the long report:

   Total 50 ms deadlines missed: 4

Notes

  • delay() is a big enemy of loop responsiveness. It directly eats into your loop() responsiveness. The CPUmeter sees this as 'busy time', but the processor isn't really doing anything helpful. For example, even if a controller is waiting for an oven to warm up, it might still be good to check if someone presses the 'CANCEL' button. It would be better to avoid delay() if at all possible.
  • It takes a while to figure out how to avoid delay() -- it's a different way of thinking of the problem. You might get some ideas from learning about "Finite State Machines". See the 'arduino-timer' library for ways to do other things while some tasks are waiting.
  • The "percent CPU load" measurement is easy to calculate, but a simplification. The meter is just scaling against the best-case loop() times the meter has seen. If loop() update rates don't vary much, the meter won't get much sense of how busy or idle the system can get. A shorter sample interval might help detect variability. It also might technically be better to scale load against 'empirical best-case loop update rate' to measure loading, but I don't have that many Arduinos :-) .