forked from Delsian/CanSerial
-
Notifications
You must be signed in to change notification settings - Fork 1
/
portnumber.c
executable file
·143 lines (123 loc) · 2.96 KB
/
portnumber.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
/*
* Port number allocation for CanSerial
*
* Copyright (C) 2019 Eug Krashtan <eug.krashtan@gmail.com>
* This file may be distributed under the terms of the GNU GPLv3 license.
*
*/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <stdint.h>
#include "cansock.h"
#include "portnumber.h"
#define CONFIG_LINE_BUFFER_SIZE 64
#define CONFIG_FILENAME "/var/tmp/canuuids.cfg"
typedef struct {
uint16_t port;
uint8_t uuid[CAN_UUID_SIZE];
} tPnKeep;
static int pn_len = 0;
static int max_pn = 0;
static tPnKeep *dict;
// power of two
static int dictsize = 2;
static void printuuid(FILE* stream, uint8_t *u)
{
int i;
for (i=0; i< CAN_UUID_SIZE; i++) {
if (i>0)
fprintf(stream, ":");
fprintf(stream, "%02X", u[i]);
}
}
static void addnum(uint16_t p, uint8_t *u)
{
if(pn_len+1 >= dictsize) {
// Increase allocated space
dictsize *= 2;
dict = realloc(dict, sizeof(tPnKeep) * dictsize);
if (!dict) {
fprintf(stderr, "realloc failed!\n");
exit(1);
}
}
// Check duplicates
for (int i=0; i<pn_len; i++) {
if (dict[i].port == p ||
memcmp(dict[i].uuid, u, CAN_UUID_SIZE) == 0) {
printf("Duplicate port %d\n", p);
return;
}
}
memcpy(dict[pn_len].uuid, u, CAN_UUID_SIZE);
dict[pn_len].port = p;
//printf("port %d ", p);
//printuuid(stdout, u);
//printf("\n");
pn_len++;
// max port for automatic allocation
if(max_pn<p) max_pn = p;
}
void PnInit(void)
{
FILE *fp;
char buf[CONFIG_LINE_BUFFER_SIZE];
dict = malloc(sizeof(tPnKeep) * dictsize);
if (!dict) {
fprintf(stderr, "malloc failed!\n");
exit(1);
}
if (fp=fopen(CONFIG_FILENAME, "r")) {
while(fgets(buf, CONFIG_LINE_BUFFER_SIZE, fp) > 0) {
if (buf[0] == '#' || strlen(buf) < 4) {
continue;
}
int p = atoi(buf);
char c[10];
uint8_t u[CAN_UUID_SIZE] = {0};
if (sscanf(buf, "%s %hhx:%hhx:%hhx:%hhx:%hhx:%hhx\n",
c, &(u[0]), &(u[1]), &(u[2]), &(u[3]), &(u[4]), &(u[5])) == 7) {
addnum(p,u);
} else {
printf("Wrong format %s\n", buf);
}
}
fclose(fp);
} else {
// First run
if ((fp=fopen(CONFIG_FILENAME, "w")) != NULL) {
fprintf(fp,"# [port] [UUID]\n");
fclose(fp);
} else {
perror(CONFIG_FILENAME);
}
}
}
void PnStore(int port, uint8_t* u)
{
FILE *fp;
if ((fp=fopen(CONFIG_FILENAME, "a")) != NULL) {
fprintf(fp,"%d ",port);
printuuid(fp, u);
fprintf(fp,"\n");
fclose(fp);
} else {
perror("Can't store config " CONFIG_FILENAME);
}
}
uint16_t PnGetNumber(uint8_t* u)
{
for (int i=0; i<pn_len; i++) {
if (memcmp(dict[i].uuid, u, CAN_UUID_SIZE) == 0) {
return dict[i].port;
}
}
// not found, keep num in dict
addnum(max_pn+1,u);
printf("Address ");
printuuid(stdout, u);
printf(" not found in config, assigned port %d\n", max_pn);
PnStore(max_pn, u);
return max_pn; // already incremented value in previous call
}