You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
136 lines
5.0 KiB
136 lines
5.0 KiB
/* libanode: the Anode C reference implementation |
|
* Copyright (C) 2009-2010 Adam Ierymenko <adam.ierymenko@gmail.com> |
|
* |
|
* This program is free software: you can redistribute it and/or modify |
|
* it under the terms of the GNU General Public License as published by |
|
* the Free Software Foundation, either version 3 of the License, or |
|
* (at your option) any later version. |
|
* |
|
* This program is distributed in the hope that it will be useful, |
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of |
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
|
* GNU General Public License for more details. |
|
* |
|
* You should have received a copy of the GNU General Public License |
|
* along with this program. If not, see <http://www.gnu.org/licenses/>. */ |
|
|
|
#include <netinet/in.h> |
|
#include <arpa/inet.h> |
|
#include "impl/misc.h" |
|
#include "impl/types.h" |
|
#include "anode.h" |
|
|
|
const AnodeNetworkAddress AnodeNetworkAddress_ANY4 = { |
|
ANODE_NETWORK_ADDRESS_IPV4, |
|
{ 0,0,0,0, 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0 } |
|
}; |
|
const AnodeNetworkAddress AnodeNetworkAddress_ANY6 = { |
|
ANODE_NETWORK_ADDRESS_IPV6, |
|
{ 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0 ,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0 } |
|
}; |
|
const AnodeNetworkAddress AnodeNetworkAddress_LOCAL4 = { |
|
ANODE_NETWORK_ADDRESS_IPV4, |
|
{ 127,0,0,1, 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0 } |
|
}; |
|
const AnodeNetworkAddress AnodeNetworkAddress_LOCAL6 = { |
|
ANODE_NETWORK_ADDRESS_IPV6, |
|
{ 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,1 ,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0 } |
|
}; |
|
|
|
int AnodeNetworkAddress_to_string(const AnodeNetworkAddress *address,char *buf,int len) |
|
{ |
|
const char *s; |
|
|
|
switch(address->type) { |
|
case ANODE_NETWORK_ADDRESS_IPV4: |
|
s = inet_ntop(AF_INET,(const void *)address->bits,buf,len); |
|
if (s) |
|
return Anode_strlen(s); |
|
else return ANODE_ERR_INVALID_ARGUMENT; |
|
break; |
|
case ANODE_NETWORK_ADDRESS_IPV6: |
|
s = inet_ntop(AF_INET6,address->bits,buf,len); |
|
if (s) |
|
return Anode_strlen(s); |
|
else return ANODE_ERR_INVALID_ARGUMENT; |
|
/* |
|
case ANODE_NETWORK_ADDRESS_ETHERNET: |
|
break; |
|
case ANODE_NETWORK_ADDRESS_USB: |
|
break; |
|
case ANODE_NETWORK_ADDRESS_BLUETOOTH: |
|
break; |
|
case ANODE_NETWORK_ADDRESS_IPC: |
|
break; |
|
case ANODE_NETWORK_ADDRESS_80211S: |
|
break; |
|
case ANODE_NETWORK_ADDRESS_SERIAL: |
|
break; |
|
*/ |
|
case ANODE_NETWORK_ADDRESS_ANODE_256_40: |
|
return AnodeAddress_to_string((const AnodeAddress *)address->bits,buf,len); |
|
default: |
|
return ANODE_ERR_ADDRESS_TYPE_NOT_SUPPORTED; |
|
} |
|
} |
|
|
|
int AnodeNetworkAddress_from_string(const char *str,AnodeNetworkAddress *address) |
|
{ |
|
unsigned int dots = Anode_count_char(str,'.'); |
|
unsigned int colons = Anode_count_char(str,':'); |
|
|
|
if ((dots == 3)&&(!colons)) { |
|
address->type = ANODE_NETWORK_ADDRESS_IPV4; |
|
if (inet_pton(AF_INET,str,address->bits) > 0) |
|
return 0; |
|
else return ANODE_ERR_INVALID_ARGUMENT; |
|
} else if ((colons)&&(!dots)) { |
|
address->type = ANODE_NETWORK_ADDRESS_IPV6; |
|
if (inet_pton(AF_INET6,str,address->bits) > 0) |
|
return 0; |
|
else return ANODE_ERR_INVALID_ARGUMENT; |
|
} else { |
|
address->type = ANODE_NETWORK_ADDRESS_ANODE_256_40; |
|
return AnodeAddress_from_string(str,(AnodeAddress *)address->bits); |
|
} |
|
} |
|
|
|
int AnodeNetworkEndpoint_from_sockaddr(const void *sockaddr,AnodeNetworkEndpoint *endpoint) |
|
{ |
|
switch(((struct sockaddr_storage *)sockaddr)->ss_family) { |
|
case AF_INET: |
|
*((uint32_t *)endpoint->address.bits) = (uint32_t)(((struct sockaddr_in *)sockaddr)->sin_addr.s_addr); |
|
endpoint->port = (int)ntohs(((struct sockaddr_in *)sockaddr)->sin_port); |
|
return 0; |
|
case AF_INET6: |
|
Anode_memcpy(endpoint->address.bits,((struct sockaddr_in6 *)sockaddr)->sin6_addr.s6_addr,16); |
|
endpoint->port = (int)ntohs(((struct sockaddr_in6 *)sockaddr)->sin6_port); |
|
return 0; |
|
default: |
|
return ANODE_ERR_INVALID_ARGUMENT; |
|
} |
|
} |
|
|
|
int AnodeNetworkEndpoint_to_sockaddr(const AnodeNetworkEndpoint *endpoint,void *sockaddr,int sockaddr_len) |
|
{ |
|
switch(endpoint->address.type) { |
|
case ANODE_NETWORK_ADDRESS_IPV4: |
|
if (sockaddr_len < (int)sizeof(struct sockaddr_in)) |
|
return ANODE_ERR_BUFFER_TOO_SMALL; |
|
Anode_zero(sockaddr,sizeof(struct sockaddr_in)); |
|
((struct sockaddr_in *)sockaddr)->sin_family = AF_INET; |
|
((struct sockaddr_in *)sockaddr)->sin_port = htons((uint16_t)endpoint->port); |
|
((struct sockaddr_in *)sockaddr)->sin_addr.s_addr = *((uint32_t *)endpoint->address.bits); |
|
return 0; |
|
case ANODE_NETWORK_ADDRESS_IPV6: |
|
if (sockaddr_len < (int)sizeof(struct sockaddr_in6)) |
|
return ANODE_ERR_BUFFER_TOO_SMALL; |
|
Anode_zero(sockaddr,sizeof(struct sockaddr_in6)); |
|
((struct sockaddr_in6 *)sockaddr)->sin6_family = AF_INET6; |
|
((struct sockaddr_in6 *)sockaddr)->sin6_port = htons((uint16_t)endpoint->port); |
|
Anode_memcpy(((struct sockaddr_in6 *)sockaddr)->sin6_addr.s6_addr,endpoint->address.bits,16); |
|
return 0; |
|
default: |
|
return ANODE_ERR_INVALID_ARGUMENT; |
|
} |
|
}
|
|
|