forked from cjo20/EdisonVoltage
-
Notifications
You must be signed in to change notification settings - Fork 0
/
voltage_server.c
168 lines (136 loc) · 3.74 KB
/
voltage_server.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
/* Author: Chris Oattes (cjo20)
This is incredibly hacky and requires fixing
*/
#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <string.h>
#include <unistd.h>
#include <stdlib.h>
#define BAT_FULL 4180
#define BAT_NORMAL 3680
#define BAT_LOW 3400
#define BAT_CRIT 3250
#define BAT_DEAD 2950
#define BAT_FULL_PERCENT 100
#define BAT_NORMAL_PERCENT 50
#define BAT_LOW_PERCENT 10
#define BAT_DEAD_PERCENT 0
float interpolatePercentage(float level, float above, float below, float above_percent, float below_percent)
{
float offset = level - below;
float range = above - below;
float fraction = offset / range;
float percentage_range = above_percent - below_percent;
float result = percentage_range * fraction;
result += below_percent;
return result;
}
float CalculatePercentage(float level)
{
if (level > BAT_FULL)
{
return 100;
}
else if (level > BAT_NORMAL)
{
return interpolatePercentage(level, BAT_FULL, BAT_NORMAL, BAT_FULL_PERCENT, BAT_NORMAL_PERCENT);
}
else if (level > BAT_LOW)
{
return interpolatePercentage(level, BAT_NORMAL, BAT_LOW, BAT_NORMAL_PERCENT, BAT_LOW_PERCENT);
}
else
{
return interpolatePercentage(level, BAT_LOW, BAT_DEAD, BAT_LOW_PERCENT, BAT_DEAD_PERCENT);
}
}
float get_voltage(float raw)
{
return raw * 4500.0f / 1024.0f;
}
int process_raw(char * str)
{
char * start = strchr(str, '=');
start += 2;
char * end = strchr(start, '\n');
*end = 0;
return atoi(start);
}
float read_raw()
{
char exec = '1';
int i = 0;
int num_samples = 0;
int total = 0;
for (i = 0; i < 5; ++i)
{
int sample = open("/sys/devices/platform/bcove_adc/basincove_gpadc/sample", O_WRONLY);
if (sample)
{
int channels = open("/sys/devices/platform/bcove_adc/basincove_gpadc/channel", O_WRONLY);
write(channels, &exec, 1);
close(channels);
write(sample, &exec, 1);
close(sample);
int results = open("/sys/devices/platform/bcove_adc/basincove_gpadc/result", O_RDONLY);
char data[1024];
read(results, &data, 1024);
close(results);
total += process_raw(data);
num_samples++;
}
}
return total / (float)num_samples;
}
int main(int argc, char * argv[])
{
char buf[1024];
buf[0] = 0;
printf("Battery voltage monitoring\n");
char * trigger_path = "/tmp/battery_trigger";
char * voltage_path = "/tmp/battery_voltage";
char * percentage_path = "/tmp/battery_percentage";
char * raw_path = "/tmp/battery_raw";
dev_t dev = 0;
unlink(trigger_path);
int status = mkfifo(trigger_path, 0777);
status = mknod(voltage_path, 0777, dev);
status = mknod(percentage_path, 0777, dev);
status = mknod(raw_path, 0777, dev);
while (buf[0] != '9')
{
char output[7];
float voltage = 0.0f;
float raw = 0;
chmod(trigger_path, S_IWUSR | S_IRUSR | S_IWGRP | S_IWOTH);
int fd = open(trigger_path, O_RDONLY);
read(fd, buf, sizeof(buf));
close(fd);
if (buf[0] == '9')
{
break;
}
printf("Got trigger\n");
raw = read_raw();
voltage = get_voltage(raw);
snprintf(output, 7, "%.0f", voltage);
fd = open(voltage_path, O_WRONLY);
write(fd, output, strlen(output) + 1);
close(fd);
snprintf(output, 7, "%.0f", CalculatePercentage(voltage));
fd = open(percentage_path, O_WRONLY);
write(fd, output, strlen(output) + 1);
close(fd);
snprintf(output, 7, "%.0f", raw);
fd = open(raw_path, O_WRONLY);
write(fd, output, strlen(output) + 1);
close(fd);
printf("Done\n");
}
unlink(trigger_path);
unlink(voltage_path);
unlink(percentage_path);
unlink(raw_path);
}