-
Notifications
You must be signed in to change notification settings - Fork 61
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Load Monitoring for RoboRIO #39
base: main
Are you sure you want to change the base?
Changes from 3 commits
3d45687
9dbafd2
51df75c
134cdcc
ffe269a
34d751d
d47c3b6
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,221 @@ | ||
package org.littletonrobotics.junction; | ||
|
||
import java.io.BufferedReader; | ||
import java.io.File; | ||
import java.io.FileReader; | ||
import java.io.IOException; | ||
|
||
// Based on work done by Teams 254 & 1736 in the casserole RioLoadMonitor | ||
|
||
public class LoadMonitor { | ||
/** | ||
* Rate of update of the load variables in milliseconds. 1s should be enough? | ||
*/ | ||
public static final int UPDATE_RATE_MS = 500; | ||
static final String CPU_LOAD_VIRTUAL_FILE = "/proc/stat"; | ||
static final String MEM_LOAD_VIRTUAL_FILE = "/proc/meminfo"; | ||
private static final String OS = System.getProperty("os.name").toLowerCase(); | ||
public static LoadMonitor instance; | ||
/** | ||
* Overall (all-cpu) load percentage (non-idle time) | ||
*/ | ||
public double totalCPULoadPct = 0; | ||
/** | ||
* Memory used percentage including cached memory | ||
*/ | ||
public double totalMemUsedPct = 0; | ||
/** | ||
* JVM memory used percentage, may have discrete jumps when garbage collection occurs or jvm dedicated | ||
* memory is resized. | ||
*/ | ||
public double totalJVMMemUsedPct = 0; | ||
double prevUserTime = 0; | ||
double prevNicedTime = 0; | ||
double prevSystemTime = 0; | ||
double prevIdleTime = 0; | ||
|
||
// Will prevent burning processor cycles if data is unreachable | ||
boolean giveUp = false; | ||
/** | ||
* Constructor. Initializes measurement system and starts a two hertz | ||
* background thread to gather load info | ||
*/ | ||
public LoadMonitor() { | ||
if(!OS.contains("nix") && !OS.contains("nux") && !OS.contains("aix")){ | ||
System.out.println("Advantage Kit LoadMonitor is only supported for the RoboRIO & other linux based " + | ||
"operating systems. LoadMonitor will not be active."); | ||
return; | ||
} | ||
|
||
//Reset give up flag | ||
giveUp = false; | ||
|
||
// Kick off monitor in new thread. | ||
// Thanks to Team 254 for an example of how to do this! | ||
Thread monitorThread = new Thread(() -> { | ||
try { | ||
while (!Thread.currentThread().isInterrupted()) { | ||
periodicUpdate(); | ||
Thread.sleep(UPDATE_RATE_MS); | ||
} | ||
} catch (Exception e) { | ||
e.printStackTrace(); | ||
} | ||
|
||
}); | ||
//Set up thread properties and start it off | ||
monitorThread.setName("Load Monitor Thread"); | ||
monitorThread.setPriority(Thread.MIN_PRIORITY); | ||
monitorThread.start(); | ||
} | ||
|
||
public static LoadMonitor getInstance() { | ||
if (instance == null) { | ||
instance = new LoadMonitor(); | ||
} | ||
return instance; | ||
} | ||
|
||
/** | ||
* Updates the loads based on info from the /proc virtual | ||
* filesystem. Should be called in the background. will be called | ||
* internally by the thread started in the constructor | ||
*/ | ||
private void periodicUpdate() { | ||
String CPUTotalLoadRawLine = ""; | ||
File file; | ||
|
||
if (!giveUp) { | ||
// CPU load parsing | ||
// Get meaningful line from CPU load virtual file | ||
file = new File(CPU_LOAD_VIRTUAL_FILE); | ||
|
||
try { | ||
BufferedReader br = new BufferedReader(new FileReader(file)); | ||
CPUTotalLoadRawLine = br.readLine(); | ||
while (CPUTotalLoadRawLine != null) { | ||
if (CPUTotalLoadRawLine.startsWith("cpu ")) { | ||
break; | ||
} | ||
CPUTotalLoadRawLine = br.readLine(); | ||
} | ||
br.close(); | ||
} catch (IOException e) { | ||
System.out.println("WARNING: cannot get raw CPU load data. Giving up future attempts to read."); | ||
e.printStackTrace(); | ||
giveUp = true; | ||
} | ||
|
||
assert CPUTotalLoadRawLine != null; | ||
String[] tokens = CPUTotalLoadRawLine.split(" "); | ||
|
||
double curUserTime = 0; | ||
double curNicedTime = 0; | ||
double curSystemTime = 0; | ||
double curIdleTime = 0; | ||
try { | ||
curUserTime = Double.parseDouble(tokens[2]); //Start at 2, because RIO parses an extra empty token at 1 | ||
curNicedTime = Double.parseDouble(tokens[3]); | ||
curSystemTime = Double.parseDouble(tokens[4]); | ||
curIdleTime = Double.parseDouble(tokens[5]); | ||
} catch (Exception e) { | ||
System.out.println("WARNING: cannot parse CPU load. Giving up future attempts to read."); | ||
e.printStackTrace(); | ||
giveUp = true; | ||
} | ||
|
||
// Calculate change in time counters since last measurement | ||
double deltaUserTime = curUserTime - prevUserTime; | ||
double deltaNicedTime = curNicedTime - prevNicedTime; | ||
double deltaSystemTime = curSystemTime - prevSystemTime; | ||
double deltaIdleTime = curIdleTime - prevIdleTime; | ||
|
||
prevUserTime = curUserTime; | ||
prevNicedTime = curNicedTime; | ||
prevSystemTime = curSystemTime; | ||
prevIdleTime = curIdleTime; | ||
|
||
// Add up totals | ||
double totalInUseTime = (deltaUserTime + deltaNicedTime + deltaSystemTime); | ||
double totalTime = totalInUseTime + deltaIdleTime; | ||
|
||
// Calculate CPU load to nearest tenth of percent | ||
totalCPULoadPct = ((double) Math.round(totalInUseTime / totalTime * 1000.0)) / 10.0; | ||
|
||
// Memory parsing and calculation | ||
String memTotalStr = ""; | ||
String memFreeStr = ""; | ||
String line; | ||
|
||
// Get meaningful line from CPU load virtual file | ||
file = new File(MEM_LOAD_VIRTUAL_FILE); | ||
try { | ||
BufferedReader br = new BufferedReader(new FileReader(file)); | ||
line = br.readLine(); | ||
while (line != null) { | ||
if (line.startsWith("MemTotal: ")) { | ||
memTotalStr = line; | ||
} else if (line.startsWith("MemFree:")) { | ||
memFreeStr = line; | ||
break; | ||
} | ||
line = br.readLine(); | ||
} | ||
br.close(); | ||
} catch (IOException e) { | ||
System.out.println("WARNING: cannot get raw memory load data. Giving up future attempts to read."); | ||
e.printStackTrace(); | ||
giveUp = true; | ||
} | ||
|
||
String[] memTotalTokens = memTotalStr.split("\\s+"); | ||
String[] memFreeTokens = memFreeStr.split("\\s+"); | ||
|
||
// Parse values from proper tokens | ||
double curTotalMem = 0; | ||
double curFreeMem = 0; | ||
try { | ||
curTotalMem = Double.parseDouble(memTotalTokens[1]); | ||
curFreeMem = Double.parseDouble(memFreeTokens[1]); | ||
} catch (Exception e) { | ||
System.out.println("WARNING: cannot parse memory load. Giving up future attempts to read."); | ||
e.printStackTrace(); | ||
giveUp = true; | ||
} | ||
|
||
totalMemUsedPct = ((double) Math.round((1.0 - curFreeMem / curTotalMem) * 1000.0)) / 10.0; | ||
} | ||
// Grab JVM memory (outside give-up loop) | ||
double jvmTotalMem = Runtime.getRuntime().totalMemory(); | ||
double jvmFreeMem = Runtime.getRuntime().freeMemory(); | ||
totalJVMMemUsedPct = (jvmTotalMem - jvmFreeMem) / (jvmTotalMem) * 100.0; | ||
} | ||
|
||
/** | ||
* Getter for load percentage on CPU. Aggregate of all cores on the system, including | ||
* both system and user processes. | ||
* | ||
* @return percentage of non-idle time, or 0 if percentage unavailable | ||
*/ | ||
public double getCPUUtilization() { | ||
return totalCPULoadPct; | ||
} | ||
|
||
/** | ||
* Getter for the load percentage on memory. Total memory utilization includes cached memory. | ||
* | ||
* @return percentage of available system RAM, or 0 if percentage unavailable. | ||
*/ | ||
public double getSystemMemoryUtilization() { | ||
return totalMemUsedPct; | ||
} | ||
|
||
/** | ||
* Getter for the load percentage on memory in the Java Virtual Machine. | ||
* | ||
* @return percentage of available JVM RAM, or 0 if percentage unavailable. | ||
*/ | ||
public double getJVMMemoryUtilization() { | ||
return totalJVMMemUsedPct; | ||
} | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
ditto