mirror of
git://git.code.sf.net/p/xtables-addons/xtables-addons
synced 2025-09-08 05:34:58 +02:00
geoip: store database in network byte order
This allows a single database to be built and distributed as a package that is accepted by both big- and little-endian hosts. Signed-off-by: Philip Prindeville <philipp@redfish-solutions.com>
This commit is contained in:

committed by
Jan Engelhardt

parent
254c6926d3
commit
b91dbd03c7
@@ -49,6 +49,38 @@ static struct option geoip_opts[] = {
|
|||||||
{NULL},
|
{NULL},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
#if __BYTE_ORDER == __LITTLE_ENDIAN
|
||||||
|
static void geoip_swap_le16(uint16_t *buf)
|
||||||
|
{
|
||||||
|
unsigned char *p = (void *)buf;
|
||||||
|
uint16_t n= p[0] + (p[1] << 8);
|
||||||
|
p[0] = (n >> 8) & 0xff;
|
||||||
|
p[1] = n & 0xff;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void geoip_swap_in6(struct in6_addr *in6)
|
||||||
|
{
|
||||||
|
geoip_swap_le16(&in6->s6_addr16[0]);
|
||||||
|
geoip_swap_le16(&in6->s6_addr16[1]);
|
||||||
|
geoip_swap_le16(&in6->s6_addr16[2]);
|
||||||
|
geoip_swap_le16(&in6->s6_addr16[3]);
|
||||||
|
geoip_swap_le16(&in6->s6_addr16[4]);
|
||||||
|
geoip_swap_le16(&in6->s6_addr16[5]);
|
||||||
|
geoip_swap_le16(&in6->s6_addr16[6]);
|
||||||
|
geoip_swap_le16(&in6->s6_addr16[7]);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void geoip_swap_le32(uint32_t *buf)
|
||||||
|
{
|
||||||
|
unsigned char *p = (void *)buf;
|
||||||
|
uint32_t n = p[0] + (p[1] << 8) + (p[2] << 16) + (p[3] << 24);
|
||||||
|
p[0] = (n >> 24) & 0xff;
|
||||||
|
p[1] = (n >> 16) & 0xff;
|
||||||
|
p[2] = (n >> 8) & 0xff;
|
||||||
|
p[3] = n & 0xff;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
static void *
|
static void *
|
||||||
geoip_get_subnets(const char *code, uint32_t *count, uint8_t nfproto)
|
geoip_get_subnets(const char *code, uint32_t *count, uint8_t nfproto)
|
||||||
{
|
{
|
||||||
@@ -56,21 +88,15 @@ geoip_get_subnets(const char *code, uint32_t *count, uint8_t nfproto)
|
|||||||
struct stat sb;
|
struct stat sb;
|
||||||
char buf[256];
|
char buf[256];
|
||||||
int fd;
|
int fd;
|
||||||
|
#if __BYTE_ORDER == __LITTLE_ENDIAN
|
||||||
|
unsigned int n;
|
||||||
|
#endif
|
||||||
|
|
||||||
/* Use simple integer vector files */
|
/* Use simple integer vector files */
|
||||||
if (nfproto == NFPROTO_IPV6) {
|
if (nfproto == NFPROTO_IPV6)
|
||||||
#if __BYTE_ORDER == _BIG_ENDIAN
|
snprintf(buf, sizeof(buf), GEOIP_DB_DIR "/%s.iv6", code);
|
||||||
snprintf(buf, sizeof(buf), GEOIP_DB_DIR "/BE/%s.iv6", code);
|
else
|
||||||
#else
|
snprintf(buf, sizeof(buf), GEOIP_DB_DIR "/%s.iv4", code);
|
||||||
snprintf(buf, sizeof(buf), GEOIP_DB_DIR "/LE/%s.iv6", code);
|
|
||||||
#endif
|
|
||||||
} else {
|
|
||||||
#if __BYTE_ORDER == _BIG_ENDIAN
|
|
||||||
snprintf(buf, sizeof(buf), GEOIP_DB_DIR "/BE/%s.iv4", code);
|
|
||||||
#else
|
|
||||||
snprintf(buf, sizeof(buf), GEOIP_DB_DIR "/LE/%s.iv4", code);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
if ((fd = open(buf, O_RDONLY)) < 0) {
|
if ((fd = open(buf, O_RDONLY)) < 0) {
|
||||||
fprintf(stderr, "Could not open %s: %s\n", buf, strerror(errno));
|
fprintf(stderr, "Could not open %s: %s\n", buf, strerror(errno));
|
||||||
@@ -98,6 +124,25 @@ geoip_get_subnets(const char *code, uint32_t *count, uint8_t nfproto)
|
|||||||
xtables_error(OTHER_PROBLEM, "geoip: insufficient memory");
|
xtables_error(OTHER_PROBLEM, "geoip: insufficient memory");
|
||||||
read(fd, subnets, sb.st_size);
|
read(fd, subnets, sb.st_size);
|
||||||
close(fd);
|
close(fd);
|
||||||
|
|
||||||
|
#if __BYTE_ORDER == __LITTLE_ENDIAN
|
||||||
|
for (n = 0; n < *count; ++n) {
|
||||||
|
switch (nfproto) {
|
||||||
|
case NFPROTO_IPV6: {
|
||||||
|
struct geoip_subnet6 *gs6 = &(((struct geoip_subnet6 *)subnets)[n]);
|
||||||
|
geoip_swap_in6(&gs6->begin);
|
||||||
|
geoip_swap_in6(&gs6->end);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case NFPROTO_IPV4: {
|
||||||
|
struct geoip_subnet4 *gs4 = &(((struct geoip_subnet4 *)subnets)[n]);
|
||||||
|
geoip_swap_le32(&gs4->begin);
|
||||||
|
geoip_swap_le32(&gs4->end);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
return subnets;
|
return subnets;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -8,51 +8,22 @@ use IO::Handle;
|
|||||||
use Text::CSV_XS; # or trade for Text::CSV
|
use Text::CSV_XS; # or trade for Text::CSV
|
||||||
use strict;
|
use strict;
|
||||||
|
|
||||||
my $le32 = pack("V", 0x10000000);
|
|
||||||
my $be32 = pack("N", 0x10000000);
|
|
||||||
my $u32 = undef;
|
|
||||||
|
|
||||||
sub wantBE { return !$u32 || $u32 eq $be32; }
|
|
||||||
sub wantLE { return !$u32 || $u32 eq $le32; }
|
|
||||||
|
|
||||||
my $csv = Text::CSV_XS->new({
|
my $csv = Text::CSV_XS->new({
|
||||||
allow_whitespace => 1,
|
allow_whitespace => 1,
|
||||||
binary => 1,
|
binary => 1,
|
||||||
eol => $/,
|
eol => $/,
|
||||||
}); # or Text::CSV
|
}); # or Text::CSV
|
||||||
my $target_dir = ".";
|
my $target_dir = ".";
|
||||||
my $native_only = 0;
|
|
||||||
|
|
||||||
&Getopt::Long::Configure(qw(bundling));
|
&Getopt::Long::Configure(qw(bundling));
|
||||||
&GetOptions(
|
&GetOptions(
|
||||||
"D=s" => \$target_dir,
|
"D=s" => \$target_dir,
|
||||||
"n" => \$native_only,
|
|
||||||
);
|
);
|
||||||
|
|
||||||
if (!-d $target_dir) {
|
if (!-d $target_dir) {
|
||||||
print STDERR "Target directory $target_dir does not exist.\n";
|
print STDERR "Target directory $target_dir does not exist.\n";
|
||||||
exit 1;
|
exit 1;
|
||||||
}
|
}
|
||||||
my @dbs = qw(LE BE);
|
|
||||||
if ($native_only) {
|
|
||||||
$u32 = pack("L", 0x10000000);
|
|
||||||
if ($u32 eq $le32) {
|
|
||||||
@dbs = qw(LE);
|
|
||||||
} elsif ($u32 eq $be32) {
|
|
||||||
@dbs = qw(BE);
|
|
||||||
} else {
|
|
||||||
print STDERRR "Cannot determine endianness.\n";
|
|
||||||
exit 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
foreach (@dbs) {
|
|
||||||
my $dir = "$target_dir/$_";
|
|
||||||
if (!-e $dir && !mkdir($dir)) {
|
|
||||||
print STDERR "Could not mkdir $dir: $!\n";
|
|
||||||
exit 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
&dump(&collect());
|
&dump(&collect());
|
||||||
|
|
||||||
@@ -96,61 +67,35 @@ sub dump
|
|||||||
sub dump_one
|
sub dump_one
|
||||||
{
|
{
|
||||||
my($iso_code, $country) = @_;
|
my($iso_code, $country) = @_;
|
||||||
my($file, $fh_le, $fh_be);
|
my($file, $fh);
|
||||||
|
|
||||||
printf "%5u IPv6 ranges for %s %s\n",
|
printf "%5u IPv6 ranges for %s %s\n",
|
||||||
scalar(@{$country->{pool_v6}}),
|
scalar(@{$country->{pool_v6}}),
|
||||||
$iso_code, $country->{name};
|
$iso_code, $country->{name};
|
||||||
|
|
||||||
if (wantLE) {
|
$file = "$target_dir/".uc($iso_code).".iv6";
|
||||||
$file = "$target_dir/LE/".uc($iso_code).".iv6";
|
if (!open($fh, "> $file")) {
|
||||||
if (!open($fh_le, "> $file")) {
|
print STDERR "Error opening $file: $!\n";
|
||||||
print STDERR "Error opening $file: $!\n";
|
exit 1;
|
||||||
exit 1;
|
|
||||||
}
|
|
||||||
foreach my $range (@{$country->{pool_v6}}) {
|
|
||||||
print $fh_le &ip6_swap($range->[0]), &ip6_swap($range->[1]);
|
|
||||||
}
|
|
||||||
close $fh_le;
|
|
||||||
}
|
}
|
||||||
if (wantBE) {
|
foreach my $range (@{$country->{pool_v6}}) {
|
||||||
$file = "$target_dir/BE/".uc($iso_code).".iv6";
|
print $fh $range->[0], $range->[1];
|
||||||
if (!open($fh_be, "> $file")) {
|
|
||||||
print STDERR "Error opening $file: $!\n";
|
|
||||||
exit 1;
|
|
||||||
}
|
|
||||||
foreach my $range (@{$country->{pool_v6}}) {
|
|
||||||
print $fh_be $range->[0], $range->[1];
|
|
||||||
}
|
|
||||||
close $fh_be;
|
|
||||||
}
|
}
|
||||||
|
close $fh;
|
||||||
|
|
||||||
printf "%5u IPv4 ranges for %s %s\n",
|
printf "%5u IPv4 ranges for %s %s\n",
|
||||||
scalar(@{$country->{pool_v4}}),
|
scalar(@{$country->{pool_v4}}),
|
||||||
$iso_code, $country->{name};
|
$iso_code, $country->{name};
|
||||||
|
|
||||||
if (wantLE) {
|
$file = "$target_dir/".uc($iso_code).".iv4";
|
||||||
$file = "$target_dir/LE/".uc($iso_code).".iv4";
|
if (!open($fh, "> $file")) {
|
||||||
if (!open($fh_le, "> $file")) {
|
print STDERR "Error opening $file: $!\n";
|
||||||
print STDERR "Error opening $file: $!\n";
|
exit 1;
|
||||||
exit 1;
|
|
||||||
}
|
|
||||||
foreach my $range (@{$country->{pool_v4}}) {
|
|
||||||
print $fh_le pack("VV", $range->[0], $range->[1]);
|
|
||||||
}
|
|
||||||
close $fh_le;
|
|
||||||
}
|
}
|
||||||
if (wantBE) {
|
foreach my $range (@{$country->{pool_v4}}) {
|
||||||
$file = "$target_dir/BE/".uc($iso_code).".iv4";
|
print $fh pack("NN", $range->[0], $range->[1]);
|
||||||
if (!open($fh_be, "> $file")) {
|
|
||||||
print STDERR "Error opening $file: $!\n";
|
|
||||||
exit 1;
|
|
||||||
}
|
|
||||||
foreach my $range (@{$country->{pool_v4}}) {
|
|
||||||
print $fh_be pack("NN", $range->[0], $range->[1]);
|
|
||||||
}
|
|
||||||
close $fh_be;
|
|
||||||
}
|
}
|
||||||
|
close $fh;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub ip6_pack
|
sub ip6_pack
|
||||||
@@ -169,7 +114,3 @@ sub ip6_pack
|
|||||||
return pack("n*", @addr);
|
return pack("n*", @addr);
|
||||||
}
|
}
|
||||||
|
|
||||||
sub ip6_swap
|
|
||||||
{
|
|
||||||
return pack("V*", unpack("N*", shift @_));
|
|
||||||
}
|
|
||||||
|
Reference in New Issue
Block a user