djdelorie / rpms / glibc

Forked from rpms/glibc 3 years ago
Clone
Carlos O'Donell a999deb
Content-Type: text/plain; charset="utf-8"
Carlos O'Donell a999deb
MIME-Version: 1.0
Carlos O'Donell a999deb
Content-Transfer-Encoding: 7bit
Carlos O'Donell a999deb
Subject: nsswitch: Add group merging support
Carlos O'Donell a999deb
From: Stephen Gallagher <sgallagh@redhat.com>
Carlos O'Donell a999deb
X-Patchwork-Id: 10289
Carlos O'Donell a999deb
Message-Id: <1452213991-6499-1-git-send-email-sgallagh@redhat.com>
Carlos O'Donell a999deb
To: libc-alpha@sourceware.org
Carlos O'Donell a999deb
Date: Thu,  7 Jan 2016 19:46:31 -0500
Carlos O'Donell a999deb
Carlos O'Donell a999deb
https://sourceware.org/glibc/wiki/Proposals/GroupMerging
Carlos O'Donell a999deb
Carlos O'Donell a999deb
== Justification ==
Carlos O'Donell a999deb
It is common today for users to rely on centrally-managed user stores for
Carlos O'Donell a999deb
handling their user accounts. However, much software existing today does
Carlos O'Donell a999deb
not have an innate understanding of such accounts. Instead, they commonly
Carlos O'Donell a999deb
rely on membership in known groups for managing access-control (for
Carlos O'Donell a999deb
example the "wheel" group on Fedora and RHEL systems or the "adm" group
Carlos O'Donell a999deb
on Debian-derived systems). In the present incarnation of nsswitch, the
Carlos O'Donell a999deb
only way to have such groups managed by a remote user store such as
Carlos O'Donell a999deb
FreeIPA or Active Directory would be to manually remove the groups from
Carlos O'Donell a999deb
/etc/group on the clients so that nsswitch would then move past nss_files
Carlos O'Donell a999deb
and into the SSSD, nss-ldap or other remote user database.
Carlos O'Donell a999deb
Carlos O'Donell a999deb
== Solution ==
Carlos O'Donell a999deb
With this patch, a new action is introduced for nsswitch:
Carlos O'Donell a999deb
NSS_ACTION_MERGE. To take advantage of it, one will add [SUCCESS=merge]
Carlos O'Donell a999deb
between two database entries in the nsswitch.conf file. When a group is
Carlos O'Donell a999deb
located in the first of the two group entries, processing will continue
Carlos O'Donell a999deb
on to the next one. If the group is also found in the next entry (and the
Carlos O'Donell a999deb
group name and GID are an exact match), the member list of the second
Carlos O'Donell a999deb
entry will be added to the group object to be returned.
Carlos O'Donell a999deb
Carlos O'Donell a999deb
== Implementation ==
Carlos O'Donell a999deb
After each DL_LOOKUP_FN() returns, the next action is checked. If the
Carlos O'Donell a999deb
function returned NSS_STATUS_SUCCESS and the next action is
Carlos O'Donell a999deb
NSS_ACTION_MERGE, a copy of the result buffer is saved for the next pass
Carlos O'Donell a999deb
through the loop. If on this next pass through the loop the database
Carlos O'Donell a999deb
returns another instance of a group matching both the group name and GID,
Carlos O'Donell a999deb
the member list is added to the previous list and it is returned as a
Carlos O'Donell a999deb
single object. If the following database does not contain the same group,
Carlos O'Donell a999deb
then the original is copied back into the destination buffer.
Carlos O'Donell a999deb
Carlos O'Donell a999deb
This patch implements merge functionality only for the group database.
Carlos O'Donell a999deb
For other databases, there is a default implementation that will return
Carlos O'Donell a999deb
the EINVAL errno if a merge is requested. The merge functionality can be
Carlos O'Donell a999deb
implemented for other databases at a later time if such is needed. Each
Carlos O'Donell a999deb
database must provide a unique implementation of the deep-copy and merge
Carlos O'Donell a999deb
functions.
Carlos O'Donell a999deb
Carlos O'Donell a999deb
If [SUCCESS=merge] is present in nsswitch.conf for a glibc version that
Carlos O'Donell a999deb
does not support it, glibc will process results up until that operation,
Carlos O'Donell a999deb
at which time it will return results if it has found them or else will
Carlos O'Donell a999deb
simply return an error. In practical terms, this ends up behaving like
Carlos O'Donell a999deb
the remainder of the nsswitch.conf line does not exist.
Carlos O'Donell a999deb
Carlos O'Donell a999deb
== Iterators ==
Carlos O'Donell a999deb
This feature does not modify the iterator functionality from its current
Carlos O'Donell a999deb
behavior. If getgrnam() or getgrgid() is called, glibc will iterate
Carlos O'Donell a999deb
through all entries in the `group` line in nsswitch.conf and display the
Carlos O'Donell a999deb
list of members without attempting to merge them. This is consistent with
Carlos O'Donell a999deb
the behavior of nss_files where if two separate lines are specified for
Carlos O'Donell a999deb
the same group in /etc/groups, getgrnam()/getgrgid() will display both.
Carlos O'Donell a999deb
Clients are already expected to handle this gracefully.
Carlos O'Donell a999deb
Carlos O'Donell a999deb
== No Premature Optimizations ==
Carlos O'Donell a999deb
The following is a list of places that might be eligible for
Carlos O'Donell a999deb
optimization, but were not overengineered for this initial contribution:
Carlos O'Donell a999deb
 * Any situation where a merge may occur will result in one malloc() of
Carlos O'Donell a999deb
   the same size as the input buffer.
Carlos O'Donell a999deb
 * Any situation where a merge does occur will result in a second
Carlos O'Donell a999deb
   malloc() to hold the list of pointers to member name strings.
Carlos O'Donell a999deb
 * The list of members is simply concatenated together and is not tested
Carlos O'Donell a999deb
   for uniqueness (which is identical to the behavior for nss_files,
Carlos O'Donell a999deb
   which will simply return identical values if they both exist on the
Carlos O'Donell a999deb
   line in the file. This could potentially be optimized to reduce space
Carlos O'Donell a999deb
   usage in the buffer, but it is both complex and computationally
Carlos O'Donell a999deb
   expensive to do so.
Carlos O'Donell a999deb
Carlos O'Donell a999deb
== Testing ==
Carlos O'Donell a999deb
I performed testing by running the getent utility against my newly-built
Carlos O'Donell a999deb
glibc and configuring /etc/nsswitch.conf with the following entry:
Carlos O'Donell a999deb
group: group:      files [SUCCESS=merge] sss
Carlos O'Donell a999deb
Carlos O'Donell a999deb
In /etc/group I included the line:
Carlos O'Donell a999deb
wheel:x:10:sgallagh
Carlos O'Donell a999deb
Carlos O'Donell a999deb
I then configured my local SSSD using the id_provider=local to respond
Carlos O'Donell a999deb
with:
Carlos O'Donell a999deb
wheel:*:10:localuser,localuser2
Carlos O'Donell a999deb
Carlos O'Donell a999deb
I then ran `getent group wheel` against the newly-built glibc in
Carlos O'Donell a999deb
multiple situations and received the expected output as described
Carlos O'Donell a999deb
above:
Carlos O'Donell a999deb
 * When SSSD was running.
Carlos O'Donell a999deb
 * When SSSD was configured in nsswitch.conf but the daemon was not
Carlos O'Donell a999deb
   running.
Carlos O'Donell a999deb
 * When SSSD was configured in nsswitch.conf but nss_sss.so.2 was not
Carlos O'Donell a999deb
   installed on the system.
Carlos O'Donell a999deb
 * When the order of 'sss' and 'files' was reversed.
Carlos O'Donell a999deb
 * All of the above with the [SUCCESS=merge] removed (to ensure no
Carlos O'Donell a999deb
   regressions).
Carlos O'Donell a999deb
 * All of the above with `getent group 10`.
Carlos O'Donell a999deb
 * All of the above with `getent group` with and without
Carlos O'Donell a999deb
   `enumerate=true` set in SSSD.
Carlos O'Donell a999deb
 * All of the above with and without nscd enabled on the system.
Carlos O'Donell a999deb
Carlos O'Donell a999deb
== NEWS ==
Carlos O'Donell a999deb
Carlos O'Donell a999deb
* A new NSS action is added to facilitate large distribution system
Carlos O'Donell a999deb
  administration.  The action, MERGE, allows remote user stores like
Carlos O'Donell a999deb
  LDAP to be merged into local user stores like /etc/groups in order
Carlos O'Donell a999deb
  to provide easy to use, updated, and managed sets of merged
Carlos O'Donell a999deb
  credentials.  The new action can be used by configuring it in
Carlos O'Donell a999deb
  /etc/nsswitch.conf:
Carlos O'Donell a999deb
  group: files [SUCCESS=merge] nis
Carlos O'Donell a999deb
  Implemented by Stephen Gallagher (Red Hat).
Carlos O'Donell a999deb
Carlos O'Donell a999deb
== ChangeLog ==
Carlos O'Donell a999deb
Carlos O'Donell a999deb
2015-12-16  Stephen Gallagher  <sgallagh@redhat.com>
Carlos O'Donell a999deb
Carlos O'Donell a999deb
	[BZ #19072]
Carlos O'Donell a999deb
	* grp/Makefile (headers): Add grp-merge.h
Carlos O'Donell a999deb
	(routines): Add grp-merge.
Carlos O'Donell a999deb
	* grp/getgrgid_r.c: Include grp-merge.h.
Carlos O'Donell a999deb
	(DEEPCOPY_FN): Define.
Carlos O'Donell a999deb
	(MERGE_FN): Define.
Carlos O'Donell a999deb
	* grp/getgrname_r.c: Include grp-merge.h.
Carlos O'Donell a999deb
	(DEEPCOPY_FN): Define.
Carlos O'Donell a999deb
	(MERGE_FN): Define.
Carlos O'Donell a999deb
	* grp/grp-merge.c: New file.
Carlos O'Donell a999deb
	* grp/grp-merge.h: New file.
Carlos O'Donell a999deb
	* manual/nss.texi (Actions in the NSS configuration): Describe
Carlos O'Donell a999deb
	return, continue, and merge.
Carlos O'Donell a999deb
	* nscd/Makefile: Add vpath to find grp-merge.c
Carlos O'Donell a999deb
	(nscd-modules): Add grp-merge.
Carlos O'Donell a999deb
	* nscd/getgrgid_r.c: Include grp/grp-merge.h.
Carlos O'Donell a999deb
	(DEEPCOPY_FN): Define.
Carlos O'Donell a999deb
	(MERGE_FN): Define.
Carlos O'Donell a999deb
	* nscd/getgrnam_r.c: Include grp/grp-merge.h.
Carlos O'Donell a999deb
	(DEEPCOPY_FN): Define.
Carlos O'Donell a999deb
	(MERGE_FN): Define.
Carlos O'Donell a999deb
	* nss/getXXbyYY_r.c [!DEEPCOPY_FN]: Define __copy_einval.
Carlos O'Donell a999deb
	[!MERGE_FN]: Define __merge_einval.
Carlos O'Donell a999deb
	(CHECK_MERGE): Define.
Carlos O'Donell a999deb
	(REENTRANT_NAME): Process merge if do_merge is true.
Carlos O'Donell a999deb
	* nss/getnssent_r.c (__nss_setent): Process NSS_ACTION_MERGE.
Carlos O'Donell a999deb
	(__nss_getent_r): Likewise.
Carlos O'Donell a999deb
	* nss/nsswitch.c (nss_parse_service_list): Likewise.
Carlos O'Donell a999deb
	* nss/nsswitch.h (lookup_actions): Define NSS_ACTION_MERGE.
Carlos O'Donell a999deb
Carlos O'Donell a999deb
Resolves BZ #19072
Carlos O'Donell a999deb
Carlos O'Donell a999deb
---
Carlos O'Donell a999deb
grp/Makefile      |   5 +-
Carlos O'Donell a999deb
 grp/getgrgid_r.c  |   3 +
Carlos O'Donell a999deb
 grp/getgrnam_r.c  |   4 ++
Carlos O'Donell a999deb
 grp/grp-merge.c   | 178 ++++++++++++++++++++++++++++++++++++++++++++++++++++++
Carlos O'Donell a999deb
 grp/grp-merge.h   |  35 +++++++++++
Carlos O'Donell a999deb
 manual/nss.texi   |  46 +++++++++++++-
Carlos O'Donell a999deb
 nscd/Makefile     |   5 +-
Carlos O'Donell a999deb
 nscd/getgrgid_r.c |   4 ++
Carlos O'Donell a999deb
 nscd/getgrnam_r.c |   4 ++
Carlos O'Donell a999deb
 nss/getXXbyYY_r.c | 105 +++++++++++++++++++++++++++++++-
Carlos O'Donell a999deb
 nss/getnssent_r.c |  34 ++++++++++-
Carlos O'Donell a999deb
 nss/nsswitch.c    |   3 +
Carlos O'Donell a999deb
 nss/nsswitch.h    |   3 +-
Carlos O'Donell a999deb
 13 files changed, 419 insertions(+), 10 deletions(-)
Carlos O'Donell a999deb
 create mode 100644 grp/grp-merge.c
Carlos O'Donell a999deb
 create mode 100644 grp/grp-merge.h
Carlos O'Donell a999deb
Carlos O'Donell a999deb
diff --git a/grp/Makefile b/grp/Makefile
Carlos O'Donell a999deb
index ed8cc2b0564f0e3842cd78f24a4e0788d659bbc4..52af992365268aae8cf8a80cd7216160b1431e84 100644
Carlos O'Donell a999deb
--- a/grp/Makefile
Carlos O'Donell a999deb
+++ b/grp/Makefile
Carlos O'Donell a999deb
@@ -20,15 +20,16 @@
Carlos O'Donell a999deb
 #
Carlos O'Donell a999deb
 subdir	:= grp
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
 include ../Makeconfig
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
-headers := grp.h
Carlos O'Donell a999deb
+headers := grp.h grp-merge.h
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
 routines := fgetgrent initgroups setgroups \
Carlos O'Donell a999deb
 	    getgrent getgrgid getgrnam putgrent \
Carlos O'Donell a999deb
-	    getgrent_r getgrgid_r getgrnam_r fgetgrent_r
Carlos O'Donell a999deb
+	    getgrent_r getgrgid_r getgrnam_r fgetgrent_r \
Carlos O'Donell a999deb
+	    grp-merge
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
 tests := testgrp tst-putgrent
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
 ifeq (yes,$(build-shared))
Carlos O'Donell a999deb
 test-srcs :=  tst_fgetgrent
Carlos O'Donell a999deb
diff --git a/grp/getgrgid_r.c b/grp/getgrgid_r.c
Carlos O'Donell a999deb
index 05d4d772d3ef0bfae8f9375387c41310885ce41a..447fa633807deec8f26d654ebeb6386a150d3a37 100644
Carlos O'Donell a999deb
--- a/grp/getgrgid_r.c
Carlos O'Donell a999deb
+++ b/grp/getgrgid_r.c
Carlos O'Donell a999deb
@@ -16,14 +16,17 @@
Carlos O'Donell a999deb
    License along with the GNU C Library; if not, see
Carlos O'Donell a999deb
    <http://www.gnu.org/licenses/>.  */
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
 #include <grp.h>
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
+#include "grp-merge.h"
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
 #define LOOKUP_TYPE	struct group
Carlos O'Donell a999deb
 #define FUNCTION_NAME	getgrgid
Carlos O'Donell a999deb
 #define DATABASE_NAME	group
Carlos O'Donell a999deb
 #define ADD_PARAMS	gid_t gid
Carlos O'Donell a999deb
 #define ADD_VARIABLES	gid
Carlos O'Donell a999deb
 #define BUFLEN		NSS_BUFLEN_GROUP
Carlos O'Donell a999deb
+#define DEEPCOPY_FN	__copy_grp
Carlos O'Donell a999deb
+#define MERGE_FN	__merge_grp
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
 #include <nss/getXXbyYY_r.c>
Carlos O'Donell a999deb
diff --git a/grp/getgrnam_r.c b/grp/getgrnam_r.c
Carlos O'Donell a999deb
index 0061cb2f7e0bd311d19775e49eb3fdd8a93447f1..c5535f4057ddfc40965f27789c34345045c8bf3b 100644
Carlos O'Donell a999deb
--- a/grp/getgrnam_r.c
Carlos O'Donell a999deb
+++ b/grp/getgrnam_r.c
Carlos O'Donell a999deb
@@ -16,13 +16,17 @@
Carlos O'Donell a999deb
    License along with the GNU C Library; if not, see
Carlos O'Donell a999deb
    <http://www.gnu.org/licenses/>.  */
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
 #include <grp.h>
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
+#include "grp-merge.h"
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
 #define LOOKUP_TYPE	struct group
Carlos O'Donell a999deb
 #define FUNCTION_NAME	getgrnam
Carlos O'Donell a999deb
 #define DATABASE_NAME	group
Carlos O'Donell a999deb
 #define ADD_PARAMS	const char *name
Carlos O'Donell a999deb
 #define ADD_VARIABLES	name
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
+#define DEEPCOPY_FN	__copy_grp
Carlos O'Donell a999deb
+#define MERGE_FN	__merge_grp
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
 #include <nss/getXXbyYY_r.c>
Carlos O'Donell a999deb
diff --git a/grp/grp-merge.c b/grp/grp-merge.c
Carlos O'Donell a999deb
new file mode 100644
Carlos O'Donell a999deb
index 0000000000000000000000000000000000000000..ca959dbfe403c89d6f3184f2b361b0c6488c9182
Carlos O'Donell a999deb
--- /dev/null
Carlos O'Donell a999deb
+++ b/grp/grp-merge.c
Carlos O'Donell a999deb
@@ -0,0 +1,178 @@
Carlos O'Donell a999deb
+/* Group merging implementation.
Carlos O'Donell a999deb
+   Copyright (C) 2016 Free Software Foundation, Inc.
Carlos O'Donell a999deb
+   This file is part of the GNU C Library.
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+   The GNU C Library is free software; you can redistribute it and/or
Carlos O'Donell a999deb
+   modify it under the terms of the GNU Lesser General Public
Carlos O'Donell a999deb
+   License as published by the Free Software Foundation; either
Carlos O'Donell a999deb
+   version 2.1 of the License, or (at your option) any later version.
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+   The GNU C Library is distributed in the hope that it will be useful,
Carlos O'Donell a999deb
+   but WITHOUT ANY WARRANTY; without even the implied warranty of
Carlos O'Donell a999deb
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Carlos O'Donell a999deb
+   Lesser General Public License for more details.
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+   You should have received a copy of the GNU Lesser General Public
Carlos O'Donell a999deb
+   License along with the GNU C Library; if not, see
Carlos O'Donell a999deb
+   <http://www.gnu.org/licenses/>.  */
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+#include <errno.h>
Carlos O'Donell a999deb
+#include <stdlib.h>
Carlos O'Donell a999deb
+#include <string.h>
Carlos O'Donell a999deb
+#include <grp.h>
Carlos O'Donell a999deb
+#include "grp-merge.h"
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+#define BUFCHECK(size)		\
Carlos O'Donell a999deb
+  do {						\
Carlos O'Donell a999deb
+    if (c + size > buflen)	\
Carlos O'Donell a999deb
+    {						\
Carlos O'Donell a999deb
+        free (members);		\
Carlos O'Donell a999deb
+        return ERANGE;		\
Carlos O'Donell a999deb
+    }						\
Carlos O'Donell a999deb
+  } while(0)
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+int
Carlos O'Donell a999deb
+__copy_grp (const struct group srcgrp, const size_t buflen,
Carlos O'Donell a999deb
+	    struct group *destgrp, char *destbuf, char **endptr)
Carlos O'Donell a999deb
+{
Carlos O'Donell a999deb
+  size_t i;
Carlos O'Donell a999deb
+  size_t c = 0;
Carlos O'Donell a999deb
+  size_t len;
Carlos O'Donell a999deb
+  size_t memcount;
Carlos O'Donell a999deb
+  char **members = NULL;
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+  /* Copy the GID.  */
Carlos O'Donell a999deb
+  destgrp->gr_gid = srcgrp.gr_gid;
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+  /* Copy the name.  */
Carlos O'Donell a999deb
+  len = strlen (srcgrp.gr_name) + 1;
Carlos O'Donell a999deb
+  BUFCHECK (len);
Carlos O'Donell a999deb
+  memcpy (&destbuf[c], srcgrp.gr_name, len);
Carlos O'Donell a999deb
+  destgrp->gr_name = &destbuf[c];
Carlos O'Donell a999deb
+  c += len;
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+  /* Copy the password.  */
Carlos O'Donell a999deb
+  len = strlen (srcgrp.gr_passwd) + 1;
Carlos O'Donell a999deb
+  BUFCHECK (len);
Carlos O'Donell a999deb
+  memcpy (&destbuf[c], srcgrp.gr_passwd, len);
Carlos O'Donell a999deb
+  destgrp->gr_passwd = &destbuf[c];
Carlos O'Donell a999deb
+  c += len;
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+  /* Count all of the members.  */
Carlos O'Donell a999deb
+  for (memcount = 0; srcgrp.gr_mem[memcount]; memcount++)
Carlos O'Donell a999deb
+    ;
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+  /* Allocate a temporary holding area for the pointers to the member
Carlos O'Donell a999deb
+     contents, including space for a NULL-terminator.  */
Carlos O'Donell a999deb
+  members = malloc (sizeof (char *) * (memcount + 1));
Carlos O'Donell a999deb
+  if (members == NULL)
Carlos O'Donell a999deb
+    return ENOMEM;
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+  /* Copy all of the group members to destbuf and add a pointer to each of
Carlos O'Donell a999deb
+     them into the 'members' array.  */
Carlos O'Donell a999deb
+  for (i = 0; srcgrp.gr_mem[i]; i++)
Carlos O'Donell a999deb
+    {
Carlos O'Donell a999deb
+      len = strlen (srcgrp.gr_mem[i]) + 1;
Carlos O'Donell a999deb
+      BUFCHECK (len);
Carlos O'Donell a999deb
+      memcpy (&destbuf[c], srcgrp.gr_mem[i], len);
Carlos O'Donell a999deb
+      members[i] = &destbuf[c];
Carlos O'Donell a999deb
+      c += len;
Carlos O'Donell a999deb
+    }
Carlos O'Donell a999deb
+  members[i] = NULL;
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+  /* Copy the pointers from the members array into the buffer and assign them
Carlos O'Donell a999deb
+     to the gr_mem member of destgrp.  */
Carlos O'Donell a999deb
+  destgrp->gr_mem = (char **) &destbuf[c];
Carlos O'Donell a999deb
+  len = sizeof (char *) * (memcount + 1);
Carlos O'Donell a999deb
+  BUFCHECK (len);
Carlos O'Donell a999deb
+  memcpy (&destbuf[c], members, len);
Carlos O'Donell a999deb
+  c += len;
Carlos O'Donell a999deb
+  free (members);
Carlos O'Donell a999deb
+  members = NULL;
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+  /* Save the count of members at the end.  */
Carlos O'Donell a999deb
+  BUFCHECK (sizeof (size_t));
Carlos O'Donell a999deb
+  memcpy (&destbuf[c], &memcount, sizeof (size_t));
Carlos O'Donell a999deb
+  c += sizeof (size_t);
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+  if (endptr)
Carlos O'Donell a999deb
+    *endptr = destbuf + c;
Carlos O'Donell a999deb
+  return 0;
Carlos O'Donell a999deb
+}
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+/* Check that the name, GID and passwd fields match, then
Carlos O'Donell a999deb
+   copy in the gr_mem array.  */
Carlos O'Donell a999deb
+int
Carlos O'Donell a999deb
+__merge_grp (struct group *savedgrp, char *savedbuf, char *savedend,
Carlos O'Donell a999deb
+	     size_t buflen, struct group *mergegrp, char *mergebuf)
Carlos O'Donell a999deb
+{
Carlos O'Donell a999deb
+  size_t c, i, len;
Carlos O'Donell a999deb
+  size_t savedmemcount;
Carlos O'Donell a999deb
+  size_t memcount;
Carlos O'Donell a999deb
+  size_t membersize;
Carlos O'Donell a999deb
+  char **members = NULL;
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+  /* We only support merging members of groups with identical names and
Carlos O'Donell a999deb
+     GID values. If we hit this case, we need to overwrite the current
Carlos O'Donell a999deb
+     buffer with the saved one (which is functionally equivalent to
Carlos O'Donell a999deb
+     treating the new lookup as NSS_STATUS NOTFOUND.  */
Carlos O'Donell a999deb
+  if (mergegrp->gr_gid != savedgrp->gr_gid
Carlos O'Donell a999deb
+      || strcmp (mergegrp->gr_name, savedgrp->gr_name))
Carlos O'Donell a999deb
+    return __copy_grp (*savedgrp, buflen, mergegrp, mergebuf, NULL);
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+  /* Get the count of group members from the last sizeof (size_t) bytes in the
Carlos O'Donell a999deb
+     mergegrp buffer.  */
Carlos O'Donell a999deb
+  savedmemcount = (size_t) *(savedend - sizeof (size_t));
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+  /* Get the count of new members to add.  */
Carlos O'Donell a999deb
+  for (memcount = 0; mergegrp->gr_mem[memcount]; memcount++)
Carlos O'Donell a999deb
+    ;
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+  /* Create a temporary array to hold the pointers to the member values from
Carlos O'Donell a999deb
+     both the saved and merge groups.  */
Carlos O'Donell a999deb
+  membersize = savedmemcount + memcount + 1;
Carlos O'Donell a999deb
+  members = malloc (sizeof (char *) * membersize);
Carlos O'Donell a999deb
+  if (members == NULL)
Carlos O'Donell a999deb
+    return ENOMEM;
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+  /* Copy in the existing member pointers from the saved group
Carlos O'Donell a999deb
+     Note: this is not NULL-terminated yet.  */
Carlos O'Donell a999deb
+  memcpy (members, savedgrp->gr_mem, sizeof (char *) * savedmemcount);
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+  /* Back up into the savedbuf until we get back to the NULL-terminator of the
Carlos O'Donell a999deb
+     group member list. (This means walking back savedmemcount + 1 (char *) pointers
Carlos O'Donell a999deb
+     and the member count value.
Carlos O'Donell a999deb
+     The value of c is going to be the used length of the buffer backed up by
Carlos O'Donell a999deb
+     the member count and further backed up by the size of the pointers.  */
Carlos O'Donell a999deb
+  c = savedend - savedbuf
Carlos O'Donell a999deb
+      - sizeof (size_t)
Carlos O'Donell a999deb
+      - sizeof (char *) * (savedmemcount + 1);
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+  /* Add all the new group members, overwriting the old NULL-terminator while
Carlos O'Donell a999deb
+     adding the new pointers to the temporary array.  */
Carlos O'Donell a999deb
+  for (i = 0; mergegrp->gr_mem[i]; i++)
Carlos O'Donell a999deb
+    {
Carlos O'Donell a999deb
+      len = strlen (mergegrp->gr_mem[i]) + 1;
Carlos O'Donell a999deb
+      BUFCHECK (len);
Carlos O'Donell a999deb
+      memcpy (&savedbuf[c], mergegrp->gr_mem[i], len);
Carlos O'Donell a999deb
+      members[savedmemcount + i] = &savedbuf[c];
Carlos O'Donell a999deb
+      c += len;
Carlos O'Donell a999deb
+    }
Carlos O'Donell a999deb
+  /* Add the NULL-terminator.  */
Carlos O'Donell a999deb
+  members[savedmemcount + memcount] = NULL;
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+  /* Copy the member array back into the buffer after the member list and free
Carlos O'Donell a999deb
+     the member array.  */
Carlos O'Donell a999deb
+  savedgrp->gr_mem = (char **) &savedbuf[c];
Carlos O'Donell a999deb
+  len = sizeof (char *) * membersize;
Carlos O'Donell a999deb
+  BUFCHECK (len);
Carlos O'Donell a999deb
+  memcpy (&savedbuf[c], members, len);
Carlos O'Donell a999deb
+  c += len;
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+  free (members);
Carlos O'Donell a999deb
+  members = NULL;
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+  /* Finally, copy the results back into mergebuf, since that's the buffer
Carlos O'Donell a999deb
+     that we were provided by the caller.  */
Carlos O'Donell a999deb
+  return __copy_grp (*savedgrp, buflen, mergegrp, mergebuf, NULL);
Carlos O'Donell a999deb
+}
Carlos O'Donell a999deb
diff --git a/grp/grp-merge.h b/grp/grp-merge.h
Carlos O'Donell a999deb
new file mode 100644
Carlos O'Donell a999deb
index 0000000000000000000000000000000000000000..59013487d0d907c76521ab504e265077937bfb5e
Carlos O'Donell a999deb
--- /dev/null
Carlos O'Donell a999deb
+++ b/grp/grp-merge.h
Carlos O'Donell a999deb
@@ -0,0 +1,35 @@
Carlos O'Donell a999deb
+/* Group merging implementation.
Carlos O'Donell a999deb
+   Copyright (C) 2016 Free Software Foundation, Inc.
Carlos O'Donell a999deb
+   This file is part of the GNU C Library.
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+   The GNU C Library is free software; you can redistribute it and/or
Carlos O'Donell a999deb
+   modify it under the terms of the GNU Lesser General Public
Carlos O'Donell a999deb
+   License as published by the Free Software Foundation; either
Carlos O'Donell a999deb
+   version 2.1 of the License, or (at your option) any later version.
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+   The GNU C Library is distributed in the hope that it will be useful,
Carlos O'Donell a999deb
+   but WITHOUT ANY WARRANTY; without even the implied warranty of
Carlos O'Donell a999deb
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Carlos O'Donell a999deb
+   Lesser General Public License for more details.
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+   You should have received a copy of the GNU Lesser General Public
Carlos O'Donell a999deb
+   License along with the GNU C Library; if not, see
Carlos O'Donell a999deb
+   <http://www.gnu.org/licenses/>.  */
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+#ifndef _GRP_MERGE_H
Carlos O'Donell a999deb
+#define _GRP_MERGE_H 1
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+#include <grp.h>
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+/* Duplicate a grp struct (and its members). When no longer needed, the
Carlos O'Donell a999deb
+   calling function must free(newbuf). */
Carlos O'Donell a999deb
+int
Carlos O'Donell a999deb
+__copy_grp (const struct group srcgrp, const size_t buflen,
Carlos O'Donell a999deb
+	    struct group *destgrp, char *destbuf, char **endptr);
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+/* Merge the member lists of two grp structs together. */
Carlos O'Donell a999deb
+int
Carlos O'Donell a999deb
+__merge_grp (struct group *savedgrp, char *savedbuf, char *savedend,
Carlos O'Donell a999deb
+	     size_t buflen, struct group *mergegrp, char *mergebuf);
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+#endif /* _GRP_MERGE_H */
Carlos O'Donell a999deb
diff --git a/manual/nss.texi b/manual/nss.texi
Carlos O'Donell a999deb
index 66dcceffe01f225f078e88dd006bb90e80c85723..95e3544bcd97995720e7154ec43df4090154bf4c 100644
Carlos O'Donell a999deb
--- a/manual/nss.texi
Carlos O'Donell a999deb
+++ b/manual/nss.texi
Carlos O'Donell a999deb
@@ -178,11 +178,11 @@ where
Carlos O'Donell a999deb
 @var{action} @result{} return | continue
Carlos O'Donell a999deb
 @end smallexample
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
 The case of the keywords is insignificant.  The @var{status}
Carlos O'Donell a999deb
 values are the results of a call to a lookup function of a specific
Carlos O'Donell a999deb
-service.  They mean
Carlos O'Donell a999deb
+service.  They mean:
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
 @ftable @samp
Carlos O'Donell a999deb
 @item success
Carlos O'Donell a999deb
 No error occurred and the wanted entry is returned.  The default action
Carlos O'Donell a999deb
 for this is @code{return}.
Carlos O'Donell a999deb
@@ -202,10 +202,54 @@ The service is temporarily unavailable.  This could mean a file is
Carlos O'Donell a999deb
 locked or a server currently cannot accept more connections.  The
Carlos O'Donell a999deb
 default action is @code{continue}.
Carlos O'Donell a999deb
 @end ftable
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
 @noindent
Carlos O'Donell a999deb
+The @var{action} values mean:
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+@ftable @samp
Carlos O'Donell a999deb
+@item return
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+If the status matches, stop the lookup process at this service
Carlos O'Donell a999deb
+specification.  If an entry is available, provide it to the application.
Carlos O'Donell a999deb
+If an error occurred, report it to the application.  In case of a prior
Carlos O'Donell a999deb
+@samp{merge} action, the data is combined with previous lookup results,
Carlos O'Donell a999deb
+as explained below.
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+@item continue
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+If the status matches, proceed with the lookup process at the next
Carlos O'Donell a999deb
+entry, discarding the result of the current lookup (and any merged
Carlos O'Donell a999deb
+data).  An exception is the @samp{initgroups} database and the
Carlos O'Donell a999deb
+@samp{success} status, where @samp{continue} acts like @code{merge}
Carlos O'Donell a999deb
+below.
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+@item merge
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+Proceed with the lookup process, retaining the current lookup result.
Carlos O'Donell a999deb
+This action is useful only with the @samp{success} status.  If a
Carlos O'Donell a999deb
+subsequent service lookup succeeds and has a matching @samp{return}
Carlos O'Donell a999deb
+specification, the results are merged, the lookup process ends, and the
Carlos O'Donell a999deb
+merged results are returned to the application.  If the following service
Carlos O'Donell a999deb
+has a matching @samp{merge} action, the lookup process continues,
Carlos O'Donell a999deb
+retaining the combined data from this and any previous lookups.
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+After a @code{merge} action, errors from subsequent lookups are ignored,
Carlos O'Donell a999deb
+and the data gathered so far will be returned.
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+The @samp{merge} only applies to the @samp{success} status.  It is
Carlos O'Donell a999deb
+currently implemented for the @samp{group} database and its group
Carlos O'Donell a999deb
+members field, @samp{gr_mem}.  If specified for other databases, it
Carlos O'Donell a999deb
+causes the lookup to fail (if the @var{status} matches).
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+When processing @samp{merge} for @samp{group} membership, the group GID
Carlos O'Donell a999deb
+and name must be identical for both entries. If only one or the other is
Carlos O'Donell a999deb
+a match, the behavior is undefined.
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+@end ftable
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+@noindent
Carlos O'Donell a999deb
 If we have a line like
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
 @smallexample
Carlos O'Donell a999deb
 ethers: nisplus [NOTFOUND=return] db files
Carlos O'Donell a999deb
 @end smallexample
Carlos O'Donell a999deb
diff --git a/nscd/Makefile b/nscd/Makefile
Carlos O'Donell a999deb
index e1a1aa92fc699aa132f7192da49f698a078e5910..3e6895573ae33221c728617f4c95bb3e8c5d5c47 100644
Carlos O'Donell a999deb
--- a/nscd/Makefile
Carlos O'Donell a999deb
+++ b/nscd/Makefile
Carlos O'Donell a999deb
@@ -29,16 +29,19 @@ aux	:= nscd_helper
Carlos O'Donell a999deb
 endif
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
 # To find xmalloc.c
Carlos O'Donell a999deb
 vpath %.c ../locale/programs
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
+# To find grp-merge.c
Carlos O'Donell a999deb
+vpath %.c ../grp
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
 nscd-modules := nscd connections pwdcache getpwnam_r getpwuid_r grpcache \
Carlos O'Donell a999deb
 		getgrnam_r getgrgid_r hstcache gethstbyad_r gethstbynm3_r \
Carlos O'Donell a999deb
 		getsrvbynm_r getsrvbypt_r servicescache \
Carlos O'Donell a999deb
 		dbg_log nscd_conf nscd_stat cache mem nscd_setup_thread \
Carlos O'Donell a999deb
 		xmalloc xstrdup aicache initgrcache gai res_hconf \
Carlos O'Donell a999deb
-		netgroupcache
Carlos O'Donell a999deb
+		netgroupcache grp-merge
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
 ifeq ($(build-nscd)$(have-thread-library),yesyes)
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
 others += nscd
Carlos O'Donell a999deb
 others-pie += nscd
Carlos O'Donell a999deb
diff --git a/nscd/getgrgid_r.c b/nscd/getgrgid_r.c
Carlos O'Donell a999deb
index fe5bda424169d56f642f125ef1f2df77a84de221..25de4a3b0b74841c44844a0541cf4d2365b22515 100644
Carlos O'Donell a999deb
--- a/nscd/getgrgid_r.c
Carlos O'Donell a999deb
+++ b/nscd/getgrgid_r.c
Carlos O'Donell a999deb
@@ -15,17 +15,21 @@
Carlos O'Donell a999deb
    You should have received a copy of the GNU General Public License
Carlos O'Donell a999deb
    along with this program; if not, see <http://www.gnu.org/licenses/>.  */
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
 #include <grp.h>
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
+#include "grp/grp-merge.h"
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
 #define LOOKUP_TYPE	struct group
Carlos O'Donell a999deb
 #define FUNCTION_NAME	getgrgid
Carlos O'Donell a999deb
 #define DATABASE_NAME	group
Carlos O'Donell a999deb
 #define ADD_PARAMS	gid_t gid
Carlos O'Donell a999deb
 #define ADD_VARIABLES	gid
Carlos O'Donell a999deb
 #define BUFLEN		NSS_BUFLEN_GROUP
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
+#define DEEPCOPY_FN	__copy_grp
Carlos O'Donell a999deb
+#define MERGE_FN	__merge_grp
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
 /* We are nscd, so we don't want to be talking to ourselves.  */
Carlos O'Donell a999deb
 #undef	USE_NSCD
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
 #include <nss/getXXbyYY_r.c>
Carlos O'Donell a999deb
diff --git a/nscd/getgrnam_r.c b/nscd/getgrnam_r.c
Carlos O'Donell a999deb
index 5ec56877f5798ca34c2e0074d5093cc22b6d58dc..386d66c5832ffee68e95195f6e34b723f41d0984 100644
Carlos O'Donell a999deb
--- a/nscd/getgrnam_r.c
Carlos O'Donell a999deb
+++ b/nscd/getgrnam_r.c
Carlos O'Donell a999deb
@@ -15,16 +15,20 @@
Carlos O'Donell a999deb
    You should have received a copy of the GNU General Public License
Carlos O'Donell a999deb
    along with this program; if not, see <http://www.gnu.org/licenses/>.  */
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
 #include <grp.h>
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
+#include "grp/grp-merge.h"
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
 #define LOOKUP_TYPE	struct group
Carlos O'Donell a999deb
 #define FUNCTION_NAME	getgrnam
Carlos O'Donell a999deb
 #define DATABASE_NAME	group
Carlos O'Donell a999deb
 #define ADD_PARAMS	const char *name
Carlos O'Donell a999deb
 #define ADD_VARIABLES	name
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
+#define DEEPCOPY_FN	__copy_grp
Carlos O'Donell a999deb
+#define MERGE_FN	__merge_grp
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
 /* We are nscd, so we don't want to be talking to ourselves.  */
Carlos O'Donell a999deb
 #undef	USE_NSCD
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
 #include <nss/getXXbyYY_r.c>
Carlos O'Donell a999deb
diff --git a/nss/getXXbyYY_r.c b/nss/getXXbyYY_r.c
Carlos O'Donell a999deb
index 198f8cfebd51be9c738f03f950f093b0855ab3cb..5f49ae8b828fce6dc30c7ccc9606319bffc52d5b 100644
Carlos O'Donell a999deb
--- a/nss/getXXbyYY_r.c
Carlos O'Donell a999deb
+++ b/nss/getXXbyYY_r.c
Carlos O'Donell a999deb
@@ -129,10 +129,52 @@
Carlos O'Donell a999deb
 # define AF_VAL af
Carlos O'Donell a999deb
 #else
Carlos O'Donell a999deb
 # define AF_VAL AF_INET
Carlos O'Donell a999deb
 #endif
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+/* Set defaults for merge functions that haven't been defined.  */
Carlos O'Donell a999deb
+#ifndef DEEPCOPY_FN
Carlos O'Donell a999deb
+static inline int
Carlos O'Donell a999deb
+__copy_einval (LOOKUP_TYPE a,
Carlos O'Donell a999deb
+	      const size_t b,
Carlos O'Donell a999deb
+	      LOOKUP_TYPE *c,
Carlos O'Donell a999deb
+	      char *d,
Carlos O'Donell a999deb
+	      char **e)
Carlos O'Donell a999deb
+{
Carlos O'Donell a999deb
+  return EINVAL;
Carlos O'Donell a999deb
+}
Carlos O'Donell a999deb
+# define DEEPCOPY_FN __copy_einval
Carlos O'Donell a999deb
+#endif
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+#ifndef MERGE_FN
Carlos O'Donell a999deb
+static inline int
Carlos O'Donell a999deb
+__merge_einval (LOOKUP_TYPE *a,
Carlos O'Donell a999deb
+	       char *b,
Carlos O'Donell a999deb
+	       char *c,
Carlos O'Donell a999deb
+	       size_t d,
Carlos O'Donell a999deb
+	       LOOKUP_TYPE *e,
Carlos O'Donell a999deb
+	       char *f)
Carlos O'Donell a999deb
+{
Carlos O'Donell a999deb
+  return EINVAL;
Carlos O'Donell a999deb
+}
Carlos O'Donell a999deb
+# define MERGE_FN __merge_einval
Carlos O'Donell a999deb
+#endif
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+#define CHECK_MERGE(err, status)	\
Carlos O'Donell a999deb
+do {					\
Carlos O'Donell a999deb
+  if (err)				\
Carlos O'Donell a999deb
+    {					\
Carlos O'Donell a999deb
+      __set_errno (err);		\
Carlos O'Donell a999deb
+      if (err == ERANGE)		\
Carlos O'Donell a999deb
+          status = NSS_STATUS_TRYAGAIN;	\
Carlos O'Donell a999deb
+      else				\
Carlos O'Donell a999deb
+          status = NSS_STATUS_UNAVAIL;	\
Carlos O'Donell a999deb
+      break;				\
Carlos O'Donell a999deb
+    }					\
Carlos O'Donell a999deb
+} while(0)
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
 /* Type of the lookup function we need here.  */
Carlos O'Donell a999deb
 typedef enum nss_status (*lookup_function) (ADD_PARAMS, LOOKUP_TYPE *, char *,
Carlos O'Donell a999deb
 					    size_t, int * H_ERRNO_PARM
Carlos O'Donell a999deb
 					    EXTRA_PARAMS);
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
@@ -150,17 +192,20 @@ INTERNAL (REENTRANT_NAME) (ADD_PARAMS, LOOKUP_TYPE *resbuf, char *buffer,
Carlos O'Donell a999deb
 {
Carlos O'Donell a999deb
   static bool startp_initialized;
Carlos O'Donell a999deb
   static service_user *startp;
Carlos O'Donell a999deb
   static lookup_function start_fct;
Carlos O'Donell a999deb
   service_user *nip;
Carlos O'Donell a999deb
+  int do_merge = 0;
Carlos O'Donell a999deb
+  LOOKUP_TYPE mergegrp;
Carlos O'Donell a999deb
+  char *mergebuf = NULL;
Carlos O'Donell a999deb
+  char *endptr = NULL;
Carlos O'Donell a999deb
   union
Carlos O'Donell a999deb
   {
Carlos O'Donell a999deb
     lookup_function l;
Carlos O'Donell a999deb
     void *ptr;
Carlos O'Donell a999deb
   } fct;
Carlos O'Donell a999deb
-
Carlos O'Donell a999deb
-  int no_more;
Carlos O'Donell a999deb
+  int no_more, err;
Carlos O'Donell a999deb
   enum nss_status status = NSS_STATUS_UNAVAIL;
Carlos O'Donell a999deb
 #ifdef USE_NSCD
Carlos O'Donell a999deb
   int nscd_status;
Carlos O'Donell a999deb
 #endif
Carlos O'Donell a999deb
 #ifdef NEED_H_ERRNO
Carlos O'Donell a999deb
@@ -276,13 +321,69 @@ INTERNAL (REENTRANT_NAME) (ADD_PARAMS, LOOKUP_TYPE *resbuf, char *buffer,
Carlos O'Donell a999deb
 	  && *h_errnop == NETDB_INTERNAL
Carlos O'Donell a999deb
 #endif
Carlos O'Donell a999deb
 	  && errno == ERANGE)
Carlos O'Donell a999deb
 	break;
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
+      if (do_merge)
Carlos O'Donell a999deb
+	{
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+	  if (status == NSS_STATUS_SUCCESS)
Carlos O'Donell a999deb
+	    {
Carlos O'Donell a999deb
+		/* The previous loop saved a buffer for merging.
Carlos O'Donell a999deb
+		   Perform the merge now.  */
Carlos O'Donell a999deb
+		err = MERGE_FN (&mergegrp, mergebuf, endptr, buflen, resbuf,
Carlos O'Donell a999deb
+				buffer);
Carlos O'Donell a999deb
+		CHECK_MERGE (err,status);
Carlos O'Donell a999deb
+		do_merge = 0;
Carlos O'Donell a999deb
+	    }
Carlos O'Donell a999deb
+	  else
Carlos O'Donell a999deb
+	    {
Carlos O'Donell a999deb
+	      /* If the result wasn't SUCCESS, copy the saved buffer back
Carlos O'Donell a999deb
+	         into the result buffer and set the status back to
Carlos O'Donell a999deb
+	         NSS_STATUS_SUCCESS to match the previous pass through the loop.
Carlos O'Donell a999deb
+	          * If the next action is CONTINUE, it will overwrite the value
Carlos O'Donell a999deb
+	            currently in the buffer and return the new value.
Carlos O'Donell a999deb
+	          * If the next action is RETURN, we'll return the previously-
Carlos O'Donell a999deb
+	            acquired values.
Carlos O'Donell a999deb
+	          * If the next action is MERGE, then it will be added to the buffer
Carlos O'Donell a999deb
+	            saved from the previous source.  */
Carlos O'Donell a999deb
+	      err = DEEPCOPY_FN (mergegrp, buflen, resbuf, buffer, NULL);
Carlos O'Donell a999deb
+	      CHECK_MERGE (err, status);
Carlos O'Donell a999deb
+	      status = NSS_STATUS_SUCCESS;
Carlos O'Donell a999deb
+	    }
Carlos O'Donell a999deb
+	}
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+      /* If we were are configured to merge this value with the next one,
Carlos O'Donell a999deb
+         save the current value of the group struct.  */
Carlos O'Donell a999deb
+      if (nss_next_action (nip, status) == NSS_ACTION_MERGE
Carlos O'Donell a999deb
+	  && status == NSS_STATUS_SUCCESS)
Carlos O'Donell a999deb
+	{
Carlos O'Donell a999deb
+	  /* Copy the current values into a buffer to be merged with the next
Carlos O'Donell a999deb
+	     set of retrieved values.  */
Carlos O'Donell a999deb
+	  if (!mergebuf)
Carlos O'Donell a999deb
+	    {
Carlos O'Donell a999deb
+	      /* Only allocate once and reuse it for as many merges as we need
Carlos O'Donell a999deb
+	         to perform.  */
Carlos O'Donell a999deb
+	      mergebuf = malloc (buflen);
Carlos O'Donell a999deb
+	      if (!mergebuf)
Carlos O'Donell a999deb
+		{
Carlos O'Donell a999deb
+		  __set_errno (ENOMEM);
Carlos O'Donell a999deb
+		  status = NSS_STATUS_UNAVAIL;
Carlos O'Donell a999deb
+		  break;
Carlos O'Donell a999deb
+		}
Carlos O'Donell a999deb
+	    }
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
+	  err = DEEPCOPY_FN (*resbuf, buflen, &mergegrp, mergebuf, &endptr);
Carlos O'Donell a999deb
+	  CHECK_MERGE (err, status);
Carlos O'Donell a999deb
+	  do_merge = 1;
Carlos O'Donell a999deb
+	}
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
       no_more = __nss_next2 (&nip, REENTRANT_NAME_STRING,
Carlos O'Donell a999deb
 			     REENTRANT2_NAME_STRING, &fct.ptr, status, 0);
Carlos O'Donell a999deb
     }
Carlos O'Donell a999deb
+  free(mergebuf);
Carlos O'Donell a999deb
+  mergebuf = NULL;
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
 #ifdef HANDLE_DIGITS_DOTS
Carlos O'Donell a999deb
 done:
Carlos O'Donell a999deb
 #endif
Carlos O'Donell a999deb
   *result = status == NSS_STATUS_SUCCESS ? resbuf : NULL;
Carlos O'Donell a999deb
diff --git a/nss/getnssent_r.c b/nss/getnssent_r.c
Carlos O'Donell a999deb
index f5b903671ca53ccad108eeb4e49ea40a45fa5cdf..c0743436f661d4d83045a6353b49291a4c0f220b 100644
Carlos O'Donell a999deb
--- a/nss/getnssent_r.c
Carlos O'Donell a999deb
+++ b/nss/getnssent_r.c
Carlos O'Donell a999deb
@@ -77,11 +77,25 @@ __nss_setent (const char *func_name, db_lookup_function lookup_fct,
Carlos O'Donell a999deb
       if (stayopen_tmp)
Carlos O'Donell a999deb
 	status = DL_CALL_FCT (fct.f, (*stayopen_tmp));
Carlos O'Donell a999deb
       else
Carlos O'Donell a999deb
 	status = DL_CALL_FCT (fct.f, (0));
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
-      no_more = __nss_next2 (nip, func_name, NULL, &fct.ptr, status, 0);
Carlos O'Donell a999deb
+      if (nss_next_action (*nip, status) == NSS_ACTION_MERGE)
Carlos O'Donell a999deb
+	{
Carlos O'Donell a999deb
+	  /* This is a special-case. When [SUCCESS=merge] is in play,
Carlos O'Donell a999deb
+	     _nss_next2() will skip to the next database.  Due to the
Carlos O'Donell a999deb
+	     implementation of that function, we can't know whether we're
Carlos O'Donell a999deb
+	     in an enumeration or an individual lookup, which behaves
Carlos O'Donell a999deb
+	     differently with regards to merging.  We'll treat SUCCESS as
Carlos O'Donell a999deb
+	     an indication to start the enumeration at this database. */
Carlos O'Donell a999deb
+	  no_more = 1;
Carlos O'Donell a999deb
+	}
Carlos O'Donell a999deb
+      else
Carlos O'Donell a999deb
+	{
Carlos O'Donell a999deb
+	  no_more = __nss_next2 (nip, func_name, NULL, &fct.ptr, status, 0);
Carlos O'Donell a999deb
+	}
Carlos O'Donell a999deb
+
Carlos O'Donell a999deb
       if (is_last_nip)
Carlos O'Donell a999deb
 	*last_nip = *nip;
Carlos O'Donell a999deb
     }
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
   if (stayopen_tmp)
Carlos O'Donell a999deb
@@ -173,12 +187,26 @@ __nss_getent_r (const char *getent_func_name,
Carlos O'Donell a999deb
 	  && errno == ERANGE)
Carlos O'Donell a999deb
 	break;
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
       do
Carlos O'Donell a999deb
 	{
Carlos O'Donell a999deb
-	  no_more = __nss_next2 (nip, getent_func_name, NULL, &fct.ptr,
Carlos O'Donell a999deb
-				 status, 0);
Carlos O'Donell a999deb
+	  if (status == NSS_STATUS_SUCCESS
Carlos O'Donell a999deb
+	      && nss_next_action (*nip, status) == NSS_ACTION_MERGE)
Carlos O'Donell a999deb
+	    {
Carlos O'Donell a999deb
+	      /* This is a special-case. When [SUCCESS=merge] is in play,
Carlos O'Donell a999deb
+	         _nss_next2() will skip to the next database.  Due to the
Carlos O'Donell a999deb
+	         implementation of that function, we can't know whether we're
Carlos O'Donell a999deb
+	         in an enumeration or an individual lookup, which behaves
Carlos O'Donell a999deb
+	         differently with regards to merging.  We'll treat SUCCESS as
Carlos O'Donell a999deb
+	         an indication to return the results here. */
Carlos O'Donell a999deb
+	      no_more = 1;
Carlos O'Donell a999deb
+	    }
Carlos O'Donell a999deb
+	  else
Carlos O'Donell a999deb
+	    {
Carlos O'Donell a999deb
+	      no_more = __nss_next2 (nip, getent_func_name, NULL, &fct.ptr,
Carlos O'Donell a999deb
+				     status, 0);
Carlos O'Donell a999deb
+	    }
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
 	  if (is_last_nip)
Carlos O'Donell a999deb
 	    *last_nip = *nip;
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
 	  if (! no_more)
Carlos O'Donell a999deb
diff --git a/nss/nsswitch.c b/nss/nsswitch.c
Carlos O'Donell a999deb
index faf9d1a0d5680aa79e88b2dfeea18da371c336fb..f8f60ba05aad9a571f928a3ea7d3b14f908ccb2d 100644
Carlos O'Donell a999deb
--- a/nss/nsswitch.c
Carlos O'Donell a999deb
+++ b/nss/nsswitch.c
Carlos O'Donell a999deb
@@ -710,10 +710,13 @@ nss_parse_service_list (const char *line)
Carlos O'Donell a999deb
 	      if (line - name == 6 && __strncasecmp (name, "RETURN", 6) == 0)
Carlos O'Donell a999deb
 		action = NSS_ACTION_RETURN;
Carlos O'Donell a999deb
 	      else if (line - name == 8
Carlos O'Donell a999deb
 		       && __strncasecmp (name, "CONTINUE", 8) == 0)
Carlos O'Donell a999deb
 		action = NSS_ACTION_CONTINUE;
Carlos O'Donell a999deb
+	      else if (line - name == 5
Carlos O'Donell a999deb
+		       && __strncasecmp (name, "MERGE", 5) == 0)
Carlos O'Donell a999deb
+		action = NSS_ACTION_MERGE;
Carlos O'Donell a999deb
 	      else
Carlos O'Donell a999deb
 		goto finish;
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
 	      if (not)
Carlos O'Donell a999deb
 		{
Carlos O'Donell a999deb
diff --git a/nss/nsswitch.h b/nss/nsswitch.h
Carlos O'Donell a999deb
index a5318fa82be43c8314807ad76de231e572e91c06..5bc2de3b1d82978102ac7a129c0ba5b7eb3cfd25 100644
Carlos O'Donell a999deb
--- a/nss/nsswitch.h
Carlos O'Donell a999deb
+++ b/nss/nsswitch.h
Carlos O'Donell a999deb
@@ -30,11 +30,12 @@
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
 /* Actions performed after lookup finished.  */
Carlos O'Donell a999deb
 typedef enum
Carlos O'Donell a999deb
 {
Carlos O'Donell a999deb
   NSS_ACTION_CONTINUE,
Carlos O'Donell a999deb
-  NSS_ACTION_RETURN
Carlos O'Donell a999deb
+  NSS_ACTION_RETURN,
Carlos O'Donell a999deb
+  NSS_ACTION_MERGE
Carlos O'Donell a999deb
 } lookup_actions;
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
 
Carlos O'Donell a999deb
 typedef struct service_library
Carlos O'Donell a999deb
 {