patty/src/ax25.c

165 lines
3.3 KiB
C
Raw Normal View History

#include <stdio.h>
2015-07-20 23:21:45 -05:00
#include <stdlib.h>
2015-07-20 22:33:59 -05:00
#include <string.h>
2015-07-20 23:21:45 -05:00
#include <errno.h>
#include <sys/select.h>
2015-07-20 22:33:59 -05:00
#include <patty/ax25.h>
enum addr_state {
ADDR_CALLSIGN,
ADDR_SSID
};
2020-06-07 02:46:12 -04:00
int patty_ax25_pton(const char *callsign,
patty_ax25_addr *addr) {
size_t len = strlen(callsign);
2020-06-07 02:46:12 -04:00
int i,
o = 0,
digits = 0;
2020-06-07 02:46:12 -04:00
int ssid = 0;
2020-06-07 02:46:12 -04:00
enum addr_state state = ADDR_CALLSIGN;
2020-06-07 02:46:12 -04:00
uint8_t c = '\0';
if (len == 0) {
goto error_invalid_callsign;
} else if (len > PATTY_AX25_ADDRSTRLEN) {
len = PATTY_AX25_ADDRSTRLEN;
}
2020-06-07 02:46:12 -04:00
for (i=0; i<len; i++) {
c = callsign[i];
switch (state) {
case ADDR_CALLSIGN:
if (o > PATTY_AX25_CALLSTRLEN) {
goto error_invalid_callsign;
}
if (c == '-') {
state = ADDR_SSID;
} else if (PATTY_AX25_ADDR_CHAR_VALID(c)) {
addr->callsign[o++] = (c & 0x7f) << 1;
} else {
goto error_invalid_callsign;
}
break;
case ADDR_SSID:
if (digits == 2) {
goto error_invalid_callsign;
}
if (c >= '0' && c <= '9') {
ssid *= 10;
ssid += c - '0';
digits++;
} else {
goto error_invalid_callsign;
}
2020-06-07 02:46:12 -04:00
}
}
2020-06-07 02:46:12 -04:00
if (c == '-') {
goto error_invalid_callsign;
}
while (o < PATTY_AX25_CALLSTRLEN) {
addr->callsign[o++] = ' ' << 1;
2020-06-07 02:46:12 -04:00
}
addr->ssid = (ssid & 0x0f) << 1;
return 0;
error_invalid_callsign:
errno = EINVAL;
2020-06-07 02:46:12 -04:00
return -1;
}
int patty_ax25_ntop(const patty_ax25_addr *addr,
char *dest,
size_t len) {
int i,
o = 0;
2020-06-07 02:46:12 -04:00
int ssid = (addr->ssid & 0x1e) >> 1;
2020-06-07 02:46:12 -04:00
for (i=0; i<PATTY_AX25_CALLSTRLEN; i++) {
2020-06-07 02:46:12 -04:00
uint8_t c;
if (o == len) {
goto error_overflow;
}
c = (addr->callsign[i] & 0xfe) >> 1;
if (c == ' ') {
2020-06-07 02:46:12 -04:00
break;
}
dest[o++] = c;
}
if (ssid) {
int formatted;
2020-06-07 02:46:12 -04:00
if ((formatted = snprintf(&dest[o], 1 + len - o, "-%d", ssid)) < 0) {
goto error_format_ssid;
} else {
o += formatted;
2020-06-07 02:46:12 -04:00
}
}
2020-06-07 02:46:12 -04:00
if (o > len) {
goto error_overflow;
2020-06-07 02:46:12 -04:00
}
dest[o] = '\0';
return 0;
error_format_ssid:
error_overflow:
errno = EOVERFLOW;
2020-06-07 02:46:12 -04:00
return -1;
}
2020-06-25 20:37:12 -04:00
static inline void hash_byte(uint32_t *hash, uint8_t c) {
*hash += c;
*hash += (*hash << 10);
*hash ^= (*hash >> 6);
}
void patty_ax25_addr_hash(uint32_t *hash, const patty_ax25_addr *addr) {
size_t i;
for (i=0; i<PATTY_AX25_CALLSTRLEN; i++) {
2020-06-25 20:37:12 -04:00
hash_byte(hash, addr->callsign[i] >> 1);
}
hash_byte(hash, PATTY_AX25_ADDR_SSID_NUMBER(addr->ssid));
2020-06-25 20:37:12 -04:00
}
size_t patty_ax25_addr_copy(void *buf,
patty_ax25_addr *addr,
uint8_t ssid_flags) {
size_t encoded = 0;
memcpy(buf, addr->callsign, sizeof(addr->callsign));
encoded += sizeof(addr->callsign);
((uint8_t *)buf)[encoded++] = ssid_flags | 0x60 | (addr->ssid & 0x1e);
return encoded;
}