/*
* \brief Generic network address definitions
* \author Stefan Kalkowski
* \date 2010-08-20
*/
/*
* Copyright (C) 2010-2017 Genode Labs GmbH
*
* This file is part of the Genode OS framework, which is distributed
* under the terms of the GNU Affero General Public License version 3.
*/
#ifndef _NET__NETADDRESS_H_
#define _NET__NETADDRESS_H_
/* Genode */
#include
#include
#include
namespace Net {
template class Network_address;
template
static inline Genode::size_t ascii_to(char const *,
Network_address &);
}
/**
* Generic form of a network address.
*/
template
struct Net::Network_address
{
Genode::uint8_t addr[LEN];
/******************
** Constructors **
******************/
Network_address(Genode::uint8_t value = 0) {
Genode::memset(&addr, value, LEN); }
Network_address(void *src) {
Genode::memcpy(&addr, src, LEN); }
/*********************
** Helper methods **
*********************/
void copy(void *dst) { Genode::memcpy(dst, addr, LEN); }
void print(Genode::Output &output) const
{
using namespace Genode;
for (unsigned i = 0; i < LEN; i++) {
if (!HEX) { Genode::print(output, (unsigned) addr[i]); }
else { Genode::print(output, Hex(addr[i], Hex::OMIT_PREFIX, Hex::PAD)); }
if (i < LEN - 1) output.out_char(DELIM);
}
}
/***************
** Operators **
***************/
bool operator==(const Network_address &other) const {
/*
* We compare from lowest address segment to highest
* one, because in a local context, the higher segments
* of two addresses normally don't distinguish.
* (e.g. in an IPv4 local subnet)
*/
for (int i = LEN-1; i >= 0; --i) {
if (addr[i] != other.addr[i])
return false;
}
return true;
}
bool operator!=(const Network_address &other) const {
return !(*this == other); }
}
__attribute__((packed));
template
Genode::size_t Net::ascii_to(char const *str, Net::Network_address &result)
{
using namespace Genode;
Net::Network_address result_buf;
size_t number_id = 0;
size_t read_len = 0;
while (1) {
/* read the current number */
size_t number_len =
ascii_to_unsigned(str, result_buf.addr[number_id],
HEX ? 16 : 10);
/* fail if there's no number */
if (!number_len) {
return 0; }
/* update read length and number index */
read_len += number_len;
number_id++;
/* if we have all numbers, fill result and return read length */
if (number_id == LEN) {
result = result_buf;
return read_len;
}
/* there are numbers left, check for the delimiter */
str += number_len;
if (*str != DELIM) {
return 0; }
/* seek to next number */
read_len++;
str++;
}
}
#endif /* _NET__NETADDRESS_H_ */