nss: Improve network number parsers (bz 32573, 32575)

Make sure that numbers never overflow uint32_t in inet_network to
properly validate octets encountered in IPv4 addresses.

Avoid malloca in NSS networks file code because /etc/networks lines
can be arbitrarily long. Instead of handcrafting the input for
inet_network by adding ".0" octets if they are missing, just left shift
the result. Also, do not accept invalid entries, but ignore the line
instead.

Reviewed-by: Adhemerval Zanella <adhemerval.zanella@linaro.org>
Signed-off-by: Tobias Stoeckmann <tobias@stoeckmann.org>
This commit is contained in:
Tobias Stoeckmann 2025-02-12 19:46:06 +01:00 committed by Adhemerval Zanella
parent 991febc2f4
commit 6a3cb6b1bd
6 changed files with 109 additions and 14 deletions

View file

@ -69,6 +69,8 @@ again:
if (*cp == 'x' || *cp == 'X')
digit = 0, base = 16, cp++;
while ((c = *cp) != 0) {
if (val > 0xff)
return (INADDR_NONE);
if (isdigit(c)) {
if (base == 8 && (c == '8' || c == '9'))
return (INADDR_NONE);

View file

@ -367,6 +367,7 @@ tests += tst-nss-files-hosts-multi
tests += tst-nss-files-hosts-getent
tests += tst-nss-files-alias-leak
tests += tst-nss-files-alias-truncated
tests += tst-nss-files-network
# tst_fgetgrent currently only works with shared libraries
test-srcs := tst_fgetgrent
ifeq ($(run-built-tests),yes)

View file

@ -42,7 +42,8 @@ LINE_PARSER
STRING_FIELD (addr, isspace, 1);
/* 'inet_network' does not add zeroes at the end if the network number
does not four byte values. We add them ourselves if necessary. */
does not contain four byte values. We shift result ourselves if
necessary. */
cp = strchr (addr, '.');
if (cp != NULL)
{
@ -56,20 +57,11 @@ LINE_PARSER
++n;
}
}
if (n < 4)
{
char *newp = (char *) alloca (strlen (addr) + (4 - n) * 2 + 1);
cp = stpcpy (newp, addr);
do
{
*cp++ = '.';
*cp++ = '0';
}
while (++n < 4);
*cp = '\0';
addr = newp;
}
result->n_net = __inet_network (addr);
if (result->n_net == INADDR_NONE)
return 0;
if (n < 4)
result->n_net <<= 8 * (4 - n);
result->n_addrtype = AF_INET;
})

View file

@ -0,0 +1,96 @@
/* Test long entries and truncated numbers in /etc/networks (bug 32573/32575).
Copyright (C) 2025 Free Software Foundation, Inc.
This file is part of the GNU C Library.
The GNU C Library is free software; you can redistribute it and/or
modify it under the terms of the GNU Lesser General Public
License as published by the Free Software Foundation; either
version 2.1 of the License, or (at your option) any later version.
The GNU C Library 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
Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public
License along with the GNU C Library; if not, see
<https://www.gnu.org/licenses/>. */
#include <netdb.h>
#include <gnu/lib-names.h>
#include <nss.h>
#include <stddef.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <support/check.h>
#include <support/check_nss.h>
#include <support/namespace.h>
#include <support/test-driver.h>
#include <support/xdlfcn.h>
#include <support/xunistd.h>
#include <sys/resource.h>
#define STACK_LIM 1048576
#define STRING_SIZE (2 * STACK_LIM)
struct support_chroot *chroot_env;
static void
prepare (int argc, char **argv)
{
int ret;
char *content;
char *entry = malloc (STRING_SIZE);
struct rlimit lim;
getrlimit (RLIMIT_STACK, &lim);
lim.rlim_cur = STACK_LIM;
setrlimit (RLIMIT_STACK, &lim);
if (entry == NULL)
{
puts ("malloc failed, cannot test");
exit (1);
}
memset (entry, 'A', STRING_SIZE);
entry[STRING_SIZE - 1] = 0;
ret = asprintf (&content, "%s\n%s\nnet3 %s\n",
"net1 x0000000000Ff.077", /* legal 255.63.0.0 */
"net2 xFF00000000.0.0.0", /* illegal */
entry /* illegal */);
if (ret == -1)
{
puts ("asprintf failed, cannot test");
exit (1);
}
free (entry);
chroot_env = support_chroot_create
((struct support_chroot_configuration)
{
.networks = content
});
}
static int
do_test (void)
{
support_become_root ();
if (!support_can_chroot ())
return EXIT_UNSUPPORTED;
__nss_configure_lookup ("networks", "files");
xdlopen (LIBNSS_FILES_SO, RTLD_NOW);
xchroot (chroot_env->path_chroot);
check_netent ("net1", getnetbyname ("net1"),
"name: net1\n"
"net: 0xff3f0000\n");
check_netent ("net2", getnetbyname ("net2"), "error: HOST_NOT_FOUND\n");
support_chroot_free (chroot_env);
return 0;
}
#define PREPARE prepare
#include <support/test-driver.c>

View file

@ -75,6 +75,7 @@ struct support_chroot_configuration
const char *hosts; /* /etc/hosts. */
const char *host_conf; /* /etc/host.conf. */
const char *aliases; /* /etc/aliases. */
const char *networks; /* /etc/networks. */
};
/* The result of the creation of a chroot. */
@ -92,6 +93,7 @@ struct support_chroot
char *path_hosts; /* /etc/hosts. */
char *path_host_conf; /* /etc/host.conf. */
char *path_aliases; /* /etc/aliases. */
char *path_networks; /* /etc/networks. */
};
/* Create a chroot environment. The returned data should be freed

View file

@ -57,6 +57,7 @@ support_chroot_create (struct support_chroot_configuration conf)
write_file (path_etc, "hosts", conf.hosts, &chroot->path_hosts);
write_file (path_etc, "host.conf", conf.host_conf, &chroot->path_host_conf);
write_file (path_etc, "aliases", conf.aliases, &chroot->path_aliases);
write_file (path_etc, "networks", conf.networks, &chroot->path_networks);
free (path_etc);
@ -79,5 +80,6 @@ support_chroot_free (struct support_chroot *chroot)
free (chroot->path_hosts);
free (chroot->path_host_conf);
free (chroot->path_aliases);
free (chroot->path_networks);
free (chroot);
}