mirror of
https://gitlab.com/sortix/sortix.git
synced 2023-02-13 20:55:38 -05:00
152 lines
3.6 KiB
C
152 lines
3.6 KiB
C
/*
|
|
* Copyright (c) 2016, 2020 Jonas 'Sortie' Termansen.
|
|
*
|
|
* Permission to use, copy, modify, and distribute this software for any
|
|
* purpose with or without fee is hereby granted, provided that the above
|
|
* copyright notice and this permission notice appear in all copies.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
|
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
|
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
|
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
|
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
|
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
*
|
|
* arpa/inet/inet_pton.c
|
|
* Convert string to network address.
|
|
*/
|
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <errno.h>
|
|
#include <arpa/inet.h>
|
|
#include <netinet/in.h>
|
|
#include <string.h>
|
|
#include <stdbool.h>
|
|
|
|
int inet_pton(int af, const char* restrict src, void* restrict dst)
|
|
{
|
|
if ( af == AF_INET )
|
|
{
|
|
unsigned char ip[4];
|
|
size_t index = 0;
|
|
for ( int i = 0; i < 4; i++ )
|
|
{
|
|
if ( i && src[index++] != '.' )
|
|
return 0;
|
|
if ( src[index] < '0' || '9' < src[index] )
|
|
return 0;
|
|
int num = src[index++] - '0';
|
|
if ( num )
|
|
{
|
|
if ( '0' <= src[index] && src[index] <= '9' )
|
|
{
|
|
num = 10 * num + src[index++] - '0';
|
|
if ( '0' <= src[index] && src[index] <= '9' )
|
|
{
|
|
num = 10 * num + src[index++] - '0';
|
|
if ( 256 <= num )
|
|
return 0;
|
|
}
|
|
}
|
|
}
|
|
ip[i] = num;
|
|
}
|
|
if ( src[index] )
|
|
return 0;
|
|
memcpy(dst, ip, sizeof(ip));
|
|
return 1;
|
|
}
|
|
else if ( af == AF_INET6 )
|
|
{
|
|
unsigned char ip[16];
|
|
size_t index = 0;
|
|
int compressed_at = -1;
|
|
int i;
|
|
for ( i = 0; i < 8; i++ )
|
|
{
|
|
bool compressing = compressed_at == -1 &&
|
|
src[index + 0] == ':' &&
|
|
src[index + 1] == ':';
|
|
if ( compressing )
|
|
{
|
|
index += 2;
|
|
compressed_at = i;
|
|
}
|
|
if ( !src[index] )
|
|
break;
|
|
if ( i && !compressing && src[index++] != ':' )
|
|
return 0;
|
|
int num = 0;
|
|
for ( int j = 0; j < 4; j++ )
|
|
{
|
|
if ( src[index] == '.' &&
|
|
((compressed_at == -1 && i == 6) ||
|
|
(0 <= compressed_at && i <= 6)) )
|
|
{
|
|
index -= j;
|
|
for ( int n = 0; n < 4; n++ )
|
|
{
|
|
if ( n && src[index++] != '.' )
|
|
return 0;
|
|
num = 0;
|
|
for ( int m = 0; m < 3; m++ )
|
|
{
|
|
if ( !m && src[index] == '0' )
|
|
{
|
|
index++;
|
|
break;
|
|
}
|
|
if ( '0' <= src[index] && src[index] <= '9' )
|
|
num = num * 10 + src[index++] - '0';
|
|
else if ( !m )
|
|
return 0;
|
|
else
|
|
break;
|
|
}
|
|
if ( 255 < num )
|
|
return 0;
|
|
ip[2 * i + n] = num;
|
|
}
|
|
i += 2;
|
|
goto done;
|
|
}
|
|
int dgt;
|
|
if ( '0' <= src[index] && src[index] <= '9' )
|
|
dgt = src[index] - '0';
|
|
else if ( 'a' <= src[index] && src[index] <= 'f' )
|
|
dgt = src[index] - 'a' + 10;
|
|
else if ( 'A' <= src[index] && src[index] <= 'F' )
|
|
dgt = src[index] - 'A' + 10;
|
|
else
|
|
{
|
|
if ( j == 0 )
|
|
return 0;
|
|
break;
|
|
}
|
|
num = num << 4 | dgt;
|
|
index++;
|
|
}
|
|
ip[2 * i + 0] = num >> 8 & 0xFF;
|
|
ip[2 * i + 1] = num >> 0 & 0xFF;
|
|
}
|
|
done:
|
|
if ( src[index] )
|
|
return 0;
|
|
if ( 0 <= compressed_at )
|
|
{
|
|
if ( i == 8 )
|
|
return 0;
|
|
memmove(ip + 16 - 2 * (i - compressed_at), ip + 2 * compressed_at,
|
|
(i - compressed_at) * 2);
|
|
memset(ip + 2 * compressed_at, 0, (8 - i) * 2);
|
|
}
|
|
else if ( i < 8 )
|
|
return 0;
|
|
memcpy(dst, ip, sizeof(ip));
|
|
return 1;
|
|
}
|
|
else
|
|
return errno = EAFNOSUPPORT, -1;
|
|
}
|