mirror of
https://github.com/diamante0018/MonitorRam.git
synced 2025-09-01 05:47:30 +00:00
Refactor (#1)
This commit is contained in:
14
src/common.c
Normal file
14
src/common.c
Normal file
@@ -0,0 +1,14 @@
|
||||
#include <time.h>
|
||||
#include <unistd.h>
|
||||
|
||||
#include "common.h"
|
||||
|
||||
// Prints the time
|
||||
void get_time_stamp(char* buf) {
|
||||
time_t ltime;
|
||||
struct tm result;
|
||||
|
||||
ltime = time(NULL);
|
||||
localtime_r(<ime, &result);
|
||||
asctime_r(&result, buf);
|
||||
}
|
8
src/common.h
Normal file
8
src/common.h
Normal file
@@ -0,0 +1,8 @@
|
||||
#ifndef COMMON_H
|
||||
#define COMMON_H
|
||||
|
||||
#define BUFFER_SIZE 4096
|
||||
|
||||
extern void get_time_stamp(char* buf);
|
||||
|
||||
#endif /* COMMON_H */
|
41
src/main.c
Normal file
41
src/main.c
Normal file
@@ -0,0 +1,41 @@
|
||||
#include <stdio.h>
|
||||
#include <signal.h>
|
||||
#include <unistd.h>
|
||||
|
||||
#include "common.h"
|
||||
#include "monitor.h"
|
||||
|
||||
volatile sig_atomic_t sigint_received = 0;
|
||||
|
||||
void sigint_handler(int a) {
|
||||
sigint_received = 1;
|
||||
(void)a;
|
||||
}
|
||||
|
||||
int main(int argc, char* argv[]) {
|
||||
if (argc < 2) {
|
||||
printf("Name of the program is missing\n");
|
||||
printf("Usage: <name of the program>\n");
|
||||
return 2;
|
||||
}
|
||||
|
||||
struct sigaction act;
|
||||
act.sa_handler = sigint_handler;
|
||||
sigaction(SIGINT, &act, NULL);
|
||||
|
||||
int result = setup(argv[1]);
|
||||
if (result) {
|
||||
return 1;
|
||||
}
|
||||
|
||||
while (!sigint_received) {
|
||||
if (analyse()) {
|
||||
return 3;
|
||||
}
|
||||
|
||||
sleep(10);
|
||||
}
|
||||
|
||||
printf("Normal shutdown\n");
|
||||
return 0;
|
||||
}
|
171
src/monitor.c
Normal file
171
src/monitor.c
Normal file
@@ -0,0 +1,171 @@
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
#include <signal.h>
|
||||
#include <errno.h>
|
||||
|
||||
#include <sys/types.h>
|
||||
|
||||
#include "common.h"
|
||||
#include "utils.h"
|
||||
#include "monitor.h"
|
||||
|
||||
#define MAX_MEMORY 1048576
|
||||
#define MAX_TRACK_SIZE 30
|
||||
|
||||
struct process_monitor {
|
||||
int id;
|
||||
int active;
|
||||
};
|
||||
|
||||
static struct process_monitor processes[MAX_TRACK_SIZE];
|
||||
|
||||
FILE* popen(const char* command, const char* type);
|
||||
int pclose(FILE* stream);
|
||||
char* strerror(int errnum);
|
||||
|
||||
static void print_generic_status_error() {
|
||||
printf("Error while parsing file\n");
|
||||
printf("Processes may be monitored incorrectly\n");
|
||||
}
|
||||
|
||||
static void check_status(FILE* file, int* curr_real_mem, int* peak_real_mem,
|
||||
int* curr_virt_mem, int* peak_virt_mem) {
|
||||
|
||||
char result[BUFFER_SIZE];
|
||||
memset(result, 0, sizeof result);
|
||||
|
||||
while (fscanf(file, " %1023s", result) == 1) {
|
||||
if (strcmp(result, "VmRSS:") == 0) {
|
||||
if (fscanf(file, " %d", curr_real_mem) == EOF) {
|
||||
print_generic_status_error();
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
if (strcmp(result, "VmHWM:") == 0) {
|
||||
if (fscanf(file, " %d", peak_real_mem) == EOF) {
|
||||
print_generic_status_error();
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
if (strcmp(result, "VmSize:") == 0) {
|
||||
if (fscanf(file, " %d", curr_virt_mem) == EOF) {
|
||||
print_generic_status_error();
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
if (strcmp(result, "VmPeak:") == 0) {
|
||||
if (fscanf(file, " %d", peak_virt_mem) == EOF) {
|
||||
print_generic_status_error();
|
||||
return;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static int is_process_wild(int curr_real_mem, int peak_real_mem,
|
||||
int curr_virt_mem, int peak_virt_mem, int id) {
|
||||
if (curr_real_mem <= MAX_MEMORY) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
// Sends a polite request to terminate
|
||||
printf("Terminating process: %d. Memory: %d\n", id, curr_real_mem);
|
||||
int result = kill(id, SIGTERM);
|
||||
|
||||
// Logs
|
||||
FILE* fp;
|
||||
fp = fopen("log.txt", "a");
|
||||
if (fp == NULL) {
|
||||
printf("Error while opening log.txt\n");
|
||||
perror("popen");
|
||||
return 1;
|
||||
}
|
||||
|
||||
char time[32];
|
||||
get_time_stamp(time);
|
||||
|
||||
fprintf(fp, "\nTimeStamp: %s\n", time);
|
||||
fprintf(fp, "WARNING: Exceeded %d KiB of Real Memory for process ID: %d\n",
|
||||
MAX_MEMORY, id);
|
||||
fprintf(fp,
|
||||
"Process ID: %d\ncurr_real_mem:%d KiB\npeak_real_mem:%d "
|
||||
"KiB\ncurr_virt_mem:%d KiB\npeak_virt_mem:%d KiB\n",
|
||||
id, curr_real_mem, peak_real_mem, curr_virt_mem, peak_virt_mem);
|
||||
|
||||
if (result != 0) {
|
||||
fprintf(fp, "Error while terminating process: %d\n", id);
|
||||
fprintf(fp, "Error message: %s\n", strerror(errno));
|
||||
}
|
||||
|
||||
fclose(fp);
|
||||
|
||||
return 1;
|
||||
}
|
||||
|
||||
int setup(const char* pid) {
|
||||
memset(&processes, 0, sizeof(struct process_monitor));
|
||||
|
||||
char* command = va("pgrep %s", pid);
|
||||
FILE* file = popen(command, "r");
|
||||
if (file == NULL) {
|
||||
perror("popen");
|
||||
return 1;
|
||||
}
|
||||
|
||||
char result[BUFFER_SIZE];
|
||||
memset(result, 0, sizeof result);
|
||||
|
||||
int i = 0;
|
||||
while (fgets(result, BUFFER_SIZE, file) && i < MAX_TRACK_SIZE) {
|
||||
printf("Process ID: %s", result);
|
||||
processes[i].id = atoi(result);
|
||||
processes[i].active = 1;
|
||||
|
||||
++i;
|
||||
}
|
||||
|
||||
if (i == 0) {
|
||||
printf("No process was found\n");
|
||||
pclose(file);
|
||||
return 1;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int analyse() {
|
||||
int i;
|
||||
FILE* status;
|
||||
int curr_real_mem = 0, peak_real_mem = 0, curr_virt_mem = 0,
|
||||
peak_virt_mem = 0;
|
||||
|
||||
for (i = 0; i < MAX_TRACK_SIZE && processes[i].active == 1; ++i) {
|
||||
char* command = va("cat /proc/%d/status", processes[i].id);
|
||||
status = popen(command, "r");
|
||||
if (status == NULL) {
|
||||
perror("popen");
|
||||
pclose(status);
|
||||
return 1;
|
||||
}
|
||||
|
||||
#ifdef _DEBUG
|
||||
printf("Checking %d\n", processes[i].id);
|
||||
#endif
|
||||
|
||||
check_status(status, &curr_real_mem, &peak_real_mem, &curr_virt_mem,
|
||||
&peak_virt_mem);
|
||||
if (is_process_wild(curr_real_mem, peak_real_mem, curr_virt_mem,
|
||||
peak_virt_mem, processes[i].id)) {
|
||||
processes[i].active = 0;
|
||||
}
|
||||
|
||||
pclose(status);
|
||||
curr_real_mem = 0, peak_real_mem = 0, curr_virt_mem = 0, peak_virt_mem = 0;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
7
src/monitor.h
Normal file
7
src/monitor.h
Normal file
@@ -0,0 +1,7 @@
|
||||
#ifndef MONITOR_H
|
||||
#define MONITOR_H
|
||||
|
||||
extern int setup(const char* pid);
|
||||
extern int analyse();
|
||||
|
||||
#endif /* MONITOR_H */
|
29
src/utils.c
Normal file
29
src/utils.c
Normal file
@@ -0,0 +1,29 @@
|
||||
#include <stdio.h>
|
||||
#include <stdarg.h>
|
||||
#include <threads.h>
|
||||
|
||||
#include "utils.h"
|
||||
|
||||
#define VA_BUFFER_COUNT 4
|
||||
#define VA_BUFFER_SIZE 4096
|
||||
|
||||
thread_local struct {
|
||||
char va_string[VA_BUFFER_COUNT][VA_BUFFER_SIZE];
|
||||
int index;
|
||||
} va_info_t;
|
||||
|
||||
char* va(const char* fmt, ...) {
|
||||
va_list ap;
|
||||
|
||||
int index = va_info_t.index;
|
||||
va_info_t.index = (va_info_t.index + 1) % VA_BUFFER_COUNT;
|
||||
char* buf = va_info_t.va_string[index];
|
||||
|
||||
va_start(ap, fmt);
|
||||
vsnprintf(buf, VA_BUFFER_SIZE, fmt, ap);
|
||||
va_end(ap);
|
||||
|
||||
buf[VA_BUFFER_SIZE - 1] = '\0';
|
||||
|
||||
return buf;
|
||||
}
|
6
src/utils.h
Normal file
6
src/utils.h
Normal file
@@ -0,0 +1,6 @@
|
||||
#ifndef UTILS_H
|
||||
#define UTILS_H
|
||||
|
||||
extern char* va(const char* fmt, ...);
|
||||
|
||||
#endif /* UTILS_H */
|
Reference in New Issue
Block a user