add Vendor TP WTP Timestamp to Echo Request
This commit is contained in:
parent
4f3fe0c339
commit
baf1ccbc73
@ -102,7 +102,8 @@ capwap_SOURCES = $(top_srcdir)/src/common/capwap.c \
|
|||||||
$(top_srcdir)/src/common/capwap_element_80211_wtpradiofailalarm.c \
|
$(top_srcdir)/src/common/capwap_element_80211_wtpradiofailalarm.c \
|
||||||
$(top_srcdir)/src/common/capwap_element_80211_wtpradioinformation.c \
|
$(top_srcdir)/src/common/capwap_element_80211_wtpradioinformation.c \
|
||||||
$(top_srcdir)/src/common/capwap_element_80211n_radioconf.c \
|
$(top_srcdir)/src/common/capwap_element_80211n_radioconf.c \
|
||||||
$(top_srcdir)/src/common/capwap_element_80211n_station_information.c
|
$(top_srcdir)/src/common/capwap_element_80211n_station_information.c \
|
||||||
|
$(top_srcdir)/src/common/capwap_element_vendor_travelping_wtp_timestamp.c
|
||||||
|
|
||||||
if DEBUG_BUILD
|
if DEBUG_BUILD
|
||||||
capwap_SOURCES += $(top_srcdir)/src/common/capwap_debug.c
|
capwap_SOURCES += $(top_srcdir)/src/common/capwap_debug.c
|
||||||
|
@ -91,8 +91,9 @@ static const struct capwap_message_elements_ops * capwap_80211_message_elements[
|
|||||||
/* */
|
/* */
|
||||||
#define element_ops(Id, Ops) [(Id) - 1] = &(Ops)
|
#define element_ops(Id, Ops) [(Id) - 1] = &(Ops)
|
||||||
static const struct capwap_message_elements_ops * capwap_vendor_travelping_message_elements[] = {
|
static const struct capwap_message_elements_ops * capwap_vendor_travelping_message_elements[] = {
|
||||||
element_ops(CAPWAP_ELEMENT_80211N_RADIO_CONF_TYPE, capwap_element_80211n_radioconf_ops),
|
element_ops(CAPWAP_ELEMENT_VENDOR_TRAVELPING_WTP_TIMESTAMP_TYPE, capwap_element_vendor_travelping_wtp_timestamp_ops),
|
||||||
element_ops(CAPWAP_ELEMENT_80211N_STATION_INFO_TYPE, capwap_element_80211n_station_info_ops)
|
element_ops(CAPWAP_ELEMENT_80211N_RADIO_CONF_TYPE, capwap_element_80211n_radioconf_ops),
|
||||||
|
element_ops(CAPWAP_ELEMENT_80211N_STATION_INFO_TYPE, capwap_element_80211n_station_info_ops)
|
||||||
};
|
};
|
||||||
#undef element_ops
|
#undef element_ops
|
||||||
|
|
||||||
|
109
src/common/capwap_element_vendor_travelping_wtp_timestamp.c
Normal file
109
src/common/capwap_element_vendor_travelping_wtp_timestamp.c
Normal file
@ -0,0 +1,109 @@
|
|||||||
|
#include "capwap.h"
|
||||||
|
#include "capwap_element.h"
|
||||||
|
|
||||||
|
/*
|
||||||
|
* 0 1 2 3
|
||||||
|
* 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
|
||||||
|
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
||||||
|
* | Second |
|
||||||
|
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
||||||
|
* | Fraction |
|
||||||
|
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
||||||
|
*
|
||||||
|
* WTP Timestamp
|
||||||
|
*
|
||||||
|
* Vendor Id: 18681 (Travelping GmbH)
|
||||||
|
* Type: 2
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
struct ntp_time_t {
|
||||||
|
uint32_t second;
|
||||||
|
uint32_t fraction;
|
||||||
|
};
|
||||||
|
|
||||||
|
static inline void convert_ntp_time_into_unix_time(struct ntp_time_t *ntp, struct timeval *tv)
|
||||||
|
{
|
||||||
|
tv->tv_sec = ntp->second - 0x83AA7E80; // the seconds from Jan 1, 1900 to Jan 1, 1970
|
||||||
|
tv->tv_usec = (uint32_t)( (double)ntp->fraction * 1.0e6 / (double)(1LL<<32) );
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void convert_unix_time_into_ntp_time(struct timeval *tv, struct ntp_time_t *ntp)
|
||||||
|
{
|
||||||
|
ntp->second = tv->tv_sec + 0x83AA7E80;
|
||||||
|
ntp->fraction = (uint32_t)( (double)(tv->tv_usec+1) * (double)(1LL<<32) * 1.0e-6 );
|
||||||
|
}
|
||||||
|
|
||||||
|
/* */
|
||||||
|
static void
|
||||||
|
capwap_vendor_travelping_wtp_timestamp_element_create(void *data,
|
||||||
|
capwap_message_elements_handle handle,
|
||||||
|
struct capwap_write_message_elements_ops *func)
|
||||||
|
{
|
||||||
|
struct capwap_vendor_travelping_wtp_timestamp_element *element =
|
||||||
|
(struct capwap_vendor_travelping_wtp_timestamp_element *)data;
|
||||||
|
struct ntp_time_t ntp;
|
||||||
|
|
||||||
|
ASSERT(data != NULL);
|
||||||
|
|
||||||
|
convert_unix_time_into_ntp_time(&element->tv, &ntp);
|
||||||
|
|
||||||
|
func->write_u32(handle, ntp.second);
|
||||||
|
func->write_u32(handle, ntp.fraction);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* */
|
||||||
|
static void *
|
||||||
|
capwap_vendor_travelping_wtp_timestamp_element_parsing(capwap_message_elements_handle handle,
|
||||||
|
struct capwap_read_message_elements_ops *func)
|
||||||
|
{
|
||||||
|
struct capwap_vendor_travelping_wtp_timestamp_element *data;
|
||||||
|
struct ntp_time_t ntp;
|
||||||
|
|
||||||
|
ASSERT(handle != NULL);
|
||||||
|
ASSERT(func != NULL);
|
||||||
|
|
||||||
|
if (func->read_ready(handle) != 8) {
|
||||||
|
log_printf(LOG_DEBUG, "Invalid Vendor Travelping WTP Timestamp element");
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* */
|
||||||
|
data = (struct capwap_vendor_travelping_wtp_timestamp_element *)
|
||||||
|
capwap_alloc(sizeof(struct capwap_vendor_travelping_wtp_timestamp_element));
|
||||||
|
|
||||||
|
/* Retrieve data */
|
||||||
|
func->read_u32(handle, &ntp.second);
|
||||||
|
func->read_u32(handle, &ntp.fraction);
|
||||||
|
|
||||||
|
convert_ntp_time_into_unix_time(&ntp, &data->tv);
|
||||||
|
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* */
|
||||||
|
static void *
|
||||||
|
capwap_vendor_travelping_wtp_timestamp_element_clone(void *data)
|
||||||
|
{
|
||||||
|
ASSERT(data != NULL);
|
||||||
|
|
||||||
|
return capwap_clone(data, sizeof(struct capwap_vendor_travelping_wtp_timestamp_element));
|
||||||
|
}
|
||||||
|
|
||||||
|
/* */
|
||||||
|
static void
|
||||||
|
capwap_vendor_travelping_wtp_timestamp_element_free(void* data)
|
||||||
|
{
|
||||||
|
ASSERT(data != NULL);
|
||||||
|
|
||||||
|
capwap_free(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* */
|
||||||
|
const struct capwap_message_elements_ops capwap_element_vendor_travelping_wtp_timestamp_ops = {
|
||||||
|
.category = CAPWAP_MESSAGE_ELEMENT_SINGLE,
|
||||||
|
.create = capwap_vendor_travelping_wtp_timestamp_element_create,
|
||||||
|
.parse = capwap_vendor_travelping_wtp_timestamp_element_parsing,
|
||||||
|
.clone = capwap_vendor_travelping_wtp_timestamp_element_clone,
|
||||||
|
.free = capwap_vendor_travelping_wtp_timestamp_element_free
|
||||||
|
};
|
19
src/common/capwap_element_vendor_travelping_wtp_timestamp.h
Normal file
19
src/common/capwap_element_vendor_travelping_wtp_timestamp.h
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
#ifndef __CAPWAP_ELEMENT_VENDOR_TRAVELPING_WTP_TIMESTAMP_HEADER__
|
||||||
|
#define __CAPWAP_ELEMENT_VENDOR_TRAVELPING_WTP_TIMESTAMP_HEADER__
|
||||||
|
|
||||||
|
#define CAPWAP_ELEMENT_VENDOR_TRAVELPING_WTP_TIMESTAMP_VENDOR CAPWAP_VENDOR_TRAVELPING_ID
|
||||||
|
#define CAPWAP_ELEMENT_VENDOR_TRAVELPING_WTP_TIMESTAMP_TYPE 2
|
||||||
|
#define CAPWAP_ELEMENT_VENDOR_TRAVELPING_WTP_TIMESTAMP \
|
||||||
|
(struct capwap_message_element_id){ \
|
||||||
|
.vendor = CAPWAP_ELEMENT_VENDOR_TRAVELPING_WTP_TIMESTAMP_VENDOR, \
|
||||||
|
.type = CAPWAP_ELEMENT_VENDOR_TRAVELPING_WTP_TIMESTAMP_TYPE \
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
struct capwap_vendor_travelping_wtp_timestamp_element {
|
||||||
|
struct timeval tv;
|
||||||
|
};
|
||||||
|
|
||||||
|
extern const struct capwap_message_elements_ops capwap_element_vendor_travelping_wtp_timestamp_ops;
|
||||||
|
|
||||||
|
#endif /* __CAPWAP_ELEMENT_VENDOR_TRAVELPING_WTP_TIMESTAMP_HEADER__ */
|
@ -3,6 +3,8 @@
|
|||||||
|
|
||||||
#define CAPWAP_VENDOR_TRAVELPING_ID 18681
|
#define CAPWAP_VENDOR_TRAVELPING_ID 18681
|
||||||
|
|
||||||
|
#include "capwap_element_vendor_travelping_wtp_timestamp.h"
|
||||||
|
|
||||||
/* draft-ietf-opsawg-capwap-extension-06 */
|
/* draft-ietf-opsawg-capwap-extension-06 */
|
||||||
#include "capwap_element_80211n_radioconf.h"
|
#include "capwap_element_80211n_radioconf.h"
|
||||||
#include "capwap_element_80211n_station_information.h"
|
#include "capwap_element_80211n_station_information.h"
|
||||||
|
@ -122,6 +122,9 @@ struct wtp_t {
|
|||||||
|
|
||||||
int sta_max_inactivity;
|
int sta_max_inactivity;
|
||||||
|
|
||||||
|
/* Echo statistics */
|
||||||
|
struct timeval echo_latency;
|
||||||
|
|
||||||
/* */
|
/* */
|
||||||
unsigned short fragmentid;
|
unsigned short fragmentid;
|
||||||
struct capwap_packet_rxmng* rxmngpacket;
|
struct capwap_packet_rxmng* rxmngpacket;
|
||||||
|
@ -16,6 +16,9 @@ static int send_echo_request(void)
|
|||||||
int result = -1;
|
int result = -1;
|
||||||
struct capwap_header_data capwapheader;
|
struct capwap_header_data capwapheader;
|
||||||
struct capwap_packet_txmng* txmngpacket;
|
struct capwap_packet_txmng* txmngpacket;
|
||||||
|
struct capwap_vendor_travelping_wtp_timestamp_element timestamp;
|
||||||
|
|
||||||
|
gettimeofday(×tamp.tv, NULL);
|
||||||
|
|
||||||
/* Build packet */
|
/* Build packet */
|
||||||
capwap_header_init(&capwapheader, CAPWAP_RADIOID_NONE, g_wtp.binding);
|
capwap_header_init(&capwapheader, CAPWAP_RADIOID_NONE, g_wtp.binding);
|
||||||
@ -23,7 +26,9 @@ static int send_echo_request(void)
|
|||||||
g_wtp.localseqnumber, g_wtp.mtu);
|
g_wtp.localseqnumber, g_wtp.mtu);
|
||||||
|
|
||||||
/* Add message element */
|
/* Add message element */
|
||||||
/* CAPWAP_ELEMENT_VENDORPAYLOAD */ /* TODO */
|
capwap_packet_txmng_add_message_element(txmngpacket,
|
||||||
|
CAPWAP_ELEMENT_VENDOR_TRAVELPING_WTP_TIMESTAMP,
|
||||||
|
×tamp);
|
||||||
|
|
||||||
/* Echo request complete, get fragment packets */
|
/* Echo request complete, get fragment packets */
|
||||||
wtp_free_reference_last_request();
|
wtp_free_reference_last_request();
|
||||||
@ -47,18 +52,34 @@ static int send_echo_request(void)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* */
|
/* */
|
||||||
static int receive_echo_response(struct capwap_parsed_packet* packet) {
|
static int receive_echo_response(struct capwap_parsed_packet* packet)
|
||||||
|
{
|
||||||
struct capwap_resultcode_element* resultcode;
|
struct capwap_resultcode_element* resultcode;
|
||||||
|
struct capwap_vendor_travelping_wtp_timestamp_element *timestamp;
|
||||||
|
|
||||||
ASSERT(packet != NULL);
|
ASSERT(packet != NULL);
|
||||||
|
|
||||||
/* Check the success of the Request */
|
/* Check the success of the Request */
|
||||||
resultcode = (struct capwap_resultcode_element*)capwap_get_message_element_data(packet, CAPWAP_ELEMENT_RESULTCODE);
|
resultcode = (struct capwap_resultcode_element *)
|
||||||
|
capwap_get_message_element_data(packet, CAPWAP_ELEMENT_RESULTCODE);
|
||||||
if (resultcode && !CAPWAP_RESULTCODE_OK(resultcode->code)) {
|
if (resultcode && !CAPWAP_RESULTCODE_OK(resultcode->code)) {
|
||||||
log_printf(LOG_WARNING, "Receive Echo Response with error: %d", (int)resultcode->code);
|
log_printf(LOG_WARNING, "Receive Echo Response with error: %d", (int)resultcode->code);
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
timestamp = (struct capwap_vendor_travelping_wtp_timestamp_element *)
|
||||||
|
capwap_get_message_element_data(packet, CAPWAP_ELEMENT_VENDOR_TRAVELPING_WTP_TIMESTAMP);
|
||||||
|
if (timestamp) {
|
||||||
|
struct timeval now;
|
||||||
|
|
||||||
|
gettimeofday(&now, NULL);
|
||||||
|
timersub(&now, ×tamp->tv, &g_wtp.echo_latency);
|
||||||
|
|
||||||
|
log_printf(LOG_DEBUG, "Echo Latency: %ld.%03ld ms",
|
||||||
|
g_wtp.echo_latency.tv_sec * 1000 + g_wtp.echo_latency.tv_usec / 1000,
|
||||||
|
g_wtp.echo_latency.tv_usec % 1000);
|
||||||
|
}
|
||||||
|
|
||||||
/* Valid packet, free request packet */
|
/* Valid packet, free request packet */
|
||||||
wtp_free_reference_last_request();
|
wtp_free_reference_last_request();
|
||||||
return 0;
|
return 0;
|
||||||
|
Loading…
Reference in New Issue
Block a user