[v2,17/23] nscd: Convert passwd client to __nscd_generic_get

Message ID dc974c5044755a818cd015e82088a649c4141c39.1774037705.git.fweimer@redhat.com (mailing list archive)
State Failed CI
Headers
Series NSS, nscd updates (for group merging and more) |

Checks

Context Check Description
redhat-pt-bot/TryBot-apply_patch success Patch applied to master at the time it was sent
linaro-tcwg-bot/tcwg_glibc_build--master-aarch64 success Build passed
linaro-tcwg-bot/tcwg_glibc_build--master-arm success Build passed
linaro-tcwg-bot/tcwg_glibc_check--master-arm fail Test failed
linaro-tcwg-bot/tcwg_glibc_check--master-aarch64 fail Test failed

Commit Message

Florian Weimer March 20, 2026, 8:42 p.m. UTC
  ---
 nscd/nscd_getpw_r.c | 165 +++++---------------------------------------
 1 file changed, 19 insertions(+), 146 deletions(-)
  

Comments

Carlos O'Donell March 24, 2026, 9:15 p.m. UTC | #1
On 3/20/26 4:42 PM, Florian Weimer wrote:

LGTM.

Reviewed-by: Carlos O'Donell <carlos@redhat.com>

> ---
>   nscd/nscd_getpw_r.c | 165 +++++---------------------------------------
>   1 file changed, 19 insertions(+), 146 deletions(-)
> 
> diff --git a/nscd/nscd_getpw_r.c b/nscd/nscd_getpw_r.c
> index 8ce1009fe0..97a752fa48 100644
> --- a/nscd/nscd_getpw_r.c
> +++ b/nscd/nscd_getpw_r.c
> @@ -29,23 +29,21 @@
>   #include <sys/un.h>
>   #include <not-cancel.h>
>   #include <_itoa.h>
> +#include <nss/nss_generic.h>
>   
>   #include "nscd-client.h"
>   #include "nscd-dbtype.h"
>   #include "nscd_proto.h"
>   
> -static int nscd_getpw_r (const char *key, size_t keylen, request_type type,
> -			 struct passwd *resultbuf, char *buffer,
> -			 size_t buflen, struct passwd **result);
> +static int nscd_getpw_r (enum nss_lookup_type lt, const void *key,
> +			 struct passwd *resultbuf, char *buffer, size_t buflen,
> +			 struct passwd **result);
>   
>   int
>   __nscd_getpwnam_r (const char *name, struct passwd *resultbuf, char *buffer,
>   		   size_t buflen, struct passwd **result)
>   {
> -  if (name == NULL)
> -    return -1;
> -
> -  return nscd_getpw_r (name, strlen (name) + 1, GETPWBYNAME, resultbuf,
> +  return nscd_getpw_r (nss_lookup_getpwnam, name, resultbuf,
>   		       buffer, buflen, result);
>   }
>   
> @@ -53,155 +51,30 @@ int
>   __nscd_getpwuid_r (uid_t uid, struct passwd *resultbuf, char *buffer,
>   		   size_t buflen, struct passwd **result)
>   {
> -  char buf[3 * sizeof (uid_t)];
> -  buf[sizeof (buf) - 1] = '\0';
> -  char *cp = _itoa_word (uid, buf + sizeof (buf) - 1, 10, 0);
> -
> -  return nscd_getpw_r (cp, buf + sizeof (buf) - cp, GETPWBYUID, resultbuf,
> +  return nscd_getpw_r (nss_lookup_getpwuid, &uid, resultbuf,
>   		       buffer, buflen, result);
>   }
>   
>   
>   static int
> -nscd_getpw_r (const char *key, size_t keylen, request_type type,
> +nscd_getpw_r (enum nss_lookup_type lt, const void *key,
>   	      struct passwd *resultbuf, char *buffer, size_t buflen,
>   	      struct passwd **result)
>   {
> -  int gc_cycle;
> -  int nretries = 0;
> -
> -  /* If the mapping is available, try to search there instead of
> -     communicating with the nscd.  */
> -  struct mapped_database *map = __nscd_get_map_ref (pwddb, &gc_cycle);
> -
> - retry:;
> -  const char *pw_name = NULL;
> -  int retval = -1;
> -  const char *recend = (const char *) ~UINTMAX_C (0);
> -  pw_response_header pw_resp;
> -
> -  if (map != NULL)
> +  void *result1;
> +  bool ok = __nscd_generic_get (lt, key, &result1);
> +  if (ok)
>       {
> -      struct datahead *found = __nscd_cache_search (type, key, keylen, map,
> -						    sizeof pw_resp);
> -      if (found != NULL)
> +      if (result1 == NULL)
>   	{
> -	  pw_name = (const char *) (&found->data[0].pwdata + 1);
> -	  pw_resp = found->data[0].pwdata;
> -	  recend = (const char *) found->data + found->recsize;
> -	  /* Now check if we can trust pw_resp fields.  If GC is
> -	     in progress, it can contain anything.  */
> -	  if (map->head->gc_cycle != gc_cycle)
> -	    {
> -	      retval = -2;
> -	      goto out;
> -	    }
> +	  *result = NULL;
> +	  return 0;
>   	}
> +      int ret = __nss_generic_copy (lt, result1, resultbuf, buffer, buflen);
> +      free (result1);
> +      if (ret == 0)
> +	*result = resultbuf;
> +      return ret;
>       }
> -
> -  int sock = -1;
> -  if (pw_name == NULL)
> -    {
> -      sock = __nscd_open_socket (key, keylen, type, &pw_resp,
> -				 sizeof (pw_resp));
> -      if (sock == -1)
> -	{
> -	  __nscd_defer_database (pwddb);
> -	  goto out;
> -	}
> -    }
> -
> -  /* No value found so far.  */
> -  *result = NULL;
> -
> -  if (__glibc_unlikely (pw_resp.found == -1))
> -    {
> -      /* The daemon does not cache this database.  */
> -      __nscd_defer_database (pwddb);
> -      goto out_close;
> -    }
> -
> -  if (pw_resp.found == 1)
> -    {
> -      /* Set the information we already have.  */
> -      resultbuf->pw_uid = pw_resp.pw_uid;
> -      resultbuf->pw_gid = pw_resp.pw_gid;
> -
> -      char *p = buffer;
> -      /* get pw_name */
> -      resultbuf->pw_name = p;
> -      p += pw_resp.pw_name_len;
> -      /* get pw_passwd */
> -      resultbuf->pw_passwd = p;
> -      p += pw_resp.pw_passwd_len;
> -      /* get pw_gecos */
> -      resultbuf->pw_gecos = p;
> -      p += pw_resp.pw_gecos_len;
> -      /* get pw_dir */
> -      resultbuf->pw_dir = p;
> -      p += pw_resp.pw_dir_len;
> -      /* get pw_pshell */
> -      resultbuf->pw_shell = p;
> -      p += pw_resp.pw_shell_len;
> -
> -      ssize_t total = p - buffer;
> -      if (__glibc_unlikely (pw_name + total > recend))
> -	goto out_close;
> -      if (__glibc_unlikely (buflen < total))
> -	{
> -	  __set_errno (ERANGE);
> -	  retval = ERANGE;
> -	  goto out_close;
> -	}
> -
> -      retval = 0;
> -      if (pw_name == NULL)
> -	{
> -	  ssize_t nbytes = __readall (sock, buffer, total);
> -
> -	  if (__glibc_unlikely (nbytes != total))
> -	    {
> -	      /* The `errno' to some value != ERANGE.  */
> -	      __set_errno (ENOENT);
> -	      retval = ENOENT;
> -	    }
> -	  else
> -	    *result = resultbuf;
> -	}
> -      else
> -	{
> -	  /* Copy the various strings.  */
> -	  memcpy (resultbuf->pw_name, pw_name, total);
> -
> -	  /* Try to detect corrupt databases.  */
> -	  if (resultbuf->pw_name[pw_resp.pw_name_len - 1] != '\0'
> -	      || resultbuf->pw_passwd[pw_resp.pw_passwd_len - 1] != '\0'
> -	      || resultbuf->pw_gecos[pw_resp.pw_gecos_len - 1] != '\0'
> -	      || resultbuf->pw_dir[pw_resp.pw_dir_len - 1] != '\0'
> -	      || resultbuf->pw_shell[pw_resp.pw_shell_len - 1] != '\0')
> -	    {
> -	      /* We cannot use the database.  */
> -	      retval = map->head->gc_cycle != gc_cycle ? -2 : -1;
> -	      goto out_close;
> -	    }
> -
> -	  *result = resultbuf;
> -	}
> -    }
> -  else
> -    {
> -      /* Set errno to 0 to indicate no error, just no found record.  */
> -      __set_errno (0);
> -      /* Even though we have not found anything, the result is zero.  */
> -      retval = 0;
> -    }
> -
> - out_close:
> -  if (sock != -1)
> -    __close_nocancel_nostatus (sock);
> - out:
> -  if (__nscd_map_ref_retry_or_drop (&map, &gc_cycle, &nretries, retval))
> -    goto retry;
> -
> -  return retval;
> +  return -1;

  59 static int
  60 nscd_getpw_r (enum nss_lookup_type lt, const void *key,
  61               struct passwd *resultbuf, char *buffer, size_t buflen,
  62               struct passwd **result)
  63 {
  64   void *result1;
  65   bool ok = __nscd_generic_get (lt, key, &result1);
  66   if (ok)
  67     {
  68       if (result1 == NULL)
  69         {
  70           *result = NULL;
  71           return 0;
  72         }
  73       int ret = __nss_generic_copy (lt, result1, resultbuf, buffer, buflen);
  74       free (result1);
  75       if (ret == 0)
  76         *result = resultbuf;
  77       return ret;
  78     }
  79   return -1;
  80 }

OK.


>   }
  

Patch

diff --git a/nscd/nscd_getpw_r.c b/nscd/nscd_getpw_r.c
index 8ce1009fe0..97a752fa48 100644
--- a/nscd/nscd_getpw_r.c
+++ b/nscd/nscd_getpw_r.c
@@ -29,23 +29,21 @@ 
 #include <sys/un.h>
 #include <not-cancel.h>
 #include <_itoa.h>
+#include <nss/nss_generic.h>
 
 #include "nscd-client.h"
 #include "nscd-dbtype.h"
 #include "nscd_proto.h"
 
-static int nscd_getpw_r (const char *key, size_t keylen, request_type type,
-			 struct passwd *resultbuf, char *buffer,
-			 size_t buflen, struct passwd **result);
+static int nscd_getpw_r (enum nss_lookup_type lt, const void *key,
+			 struct passwd *resultbuf, char *buffer, size_t buflen,
+			 struct passwd **result);
 
 int
 __nscd_getpwnam_r (const char *name, struct passwd *resultbuf, char *buffer,
 		   size_t buflen, struct passwd **result)
 {
-  if (name == NULL)
-    return -1;
-
-  return nscd_getpw_r (name, strlen (name) + 1, GETPWBYNAME, resultbuf,
+  return nscd_getpw_r (nss_lookup_getpwnam, name, resultbuf,
 		       buffer, buflen, result);
 }
 
@@ -53,155 +51,30 @@  int
 __nscd_getpwuid_r (uid_t uid, struct passwd *resultbuf, char *buffer,
 		   size_t buflen, struct passwd **result)
 {
-  char buf[3 * sizeof (uid_t)];
-  buf[sizeof (buf) - 1] = '\0';
-  char *cp = _itoa_word (uid, buf + sizeof (buf) - 1, 10, 0);
-
-  return nscd_getpw_r (cp, buf + sizeof (buf) - cp, GETPWBYUID, resultbuf,
+  return nscd_getpw_r (nss_lookup_getpwuid, &uid, resultbuf,
 		       buffer, buflen, result);
 }
 
 
 static int
-nscd_getpw_r (const char *key, size_t keylen, request_type type,
+nscd_getpw_r (enum nss_lookup_type lt, const void *key,
 	      struct passwd *resultbuf, char *buffer, size_t buflen,
 	      struct passwd **result)
 {
-  int gc_cycle;
-  int nretries = 0;
-
-  /* If the mapping is available, try to search there instead of
-     communicating with the nscd.  */
-  struct mapped_database *map = __nscd_get_map_ref (pwddb, &gc_cycle);
-
- retry:;
-  const char *pw_name = NULL;
-  int retval = -1;
-  const char *recend = (const char *) ~UINTMAX_C (0);
-  pw_response_header pw_resp;
-
-  if (map != NULL)
+  void *result1;
+  bool ok = __nscd_generic_get (lt, key, &result1);
+  if (ok)
     {
-      struct datahead *found = __nscd_cache_search (type, key, keylen, map,
-						    sizeof pw_resp);
-      if (found != NULL)
+      if (result1 == NULL)
 	{
-	  pw_name = (const char *) (&found->data[0].pwdata + 1);
-	  pw_resp = found->data[0].pwdata;
-	  recend = (const char *) found->data + found->recsize;
-	  /* Now check if we can trust pw_resp fields.  If GC is
-	     in progress, it can contain anything.  */
-	  if (map->head->gc_cycle != gc_cycle)
-	    {
-	      retval = -2;
-	      goto out;
-	    }
+	  *result = NULL;
+	  return 0;
 	}
+      int ret = __nss_generic_copy (lt, result1, resultbuf, buffer, buflen);
+      free (result1);
+      if (ret == 0)
+	*result = resultbuf;
+      return ret;
     }
-
-  int sock = -1;
-  if (pw_name == NULL)
-    {
-      sock = __nscd_open_socket (key, keylen, type, &pw_resp,
-				 sizeof (pw_resp));
-      if (sock == -1)
-	{
-	  __nscd_defer_database (pwddb);
-	  goto out;
-	}
-    }
-
-  /* No value found so far.  */
-  *result = NULL;
-
-  if (__glibc_unlikely (pw_resp.found == -1))
-    {
-      /* The daemon does not cache this database.  */
-      __nscd_defer_database (pwddb);
-      goto out_close;
-    }
-
-  if (pw_resp.found == 1)
-    {
-      /* Set the information we already have.  */
-      resultbuf->pw_uid = pw_resp.pw_uid;
-      resultbuf->pw_gid = pw_resp.pw_gid;
-
-      char *p = buffer;
-      /* get pw_name */
-      resultbuf->pw_name = p;
-      p += pw_resp.pw_name_len;
-      /* get pw_passwd */
-      resultbuf->pw_passwd = p;
-      p += pw_resp.pw_passwd_len;
-      /* get pw_gecos */
-      resultbuf->pw_gecos = p;
-      p += pw_resp.pw_gecos_len;
-      /* get pw_dir */
-      resultbuf->pw_dir = p;
-      p += pw_resp.pw_dir_len;
-      /* get pw_pshell */
-      resultbuf->pw_shell = p;
-      p += pw_resp.pw_shell_len;
-
-      ssize_t total = p - buffer;
-      if (__glibc_unlikely (pw_name + total > recend))
-	goto out_close;
-      if (__glibc_unlikely (buflen < total))
-	{
-	  __set_errno (ERANGE);
-	  retval = ERANGE;
-	  goto out_close;
-	}
-
-      retval = 0;
-      if (pw_name == NULL)
-	{
-	  ssize_t nbytes = __readall (sock, buffer, total);
-
-	  if (__glibc_unlikely (nbytes != total))
-	    {
-	      /* The `errno' to some value != ERANGE.  */
-	      __set_errno (ENOENT);
-	      retval = ENOENT;
-	    }
-	  else
-	    *result = resultbuf;
-	}
-      else
-	{
-	  /* Copy the various strings.  */
-	  memcpy (resultbuf->pw_name, pw_name, total);
-
-	  /* Try to detect corrupt databases.  */
-	  if (resultbuf->pw_name[pw_resp.pw_name_len - 1] != '\0'
-	      || resultbuf->pw_passwd[pw_resp.pw_passwd_len - 1] != '\0'
-	      || resultbuf->pw_gecos[pw_resp.pw_gecos_len - 1] != '\0'
-	      || resultbuf->pw_dir[pw_resp.pw_dir_len - 1] != '\0'
-	      || resultbuf->pw_shell[pw_resp.pw_shell_len - 1] != '\0')
-	    {
-	      /* We cannot use the database.  */
-	      retval = map->head->gc_cycle != gc_cycle ? -2 : -1;
-	      goto out_close;
-	    }
-
-	  *result = resultbuf;
-	}
-    }
-  else
-    {
-      /* Set errno to 0 to indicate no error, just no found record.  */
-      __set_errno (0);
-      /* Even though we have not found anything, the result is zero.  */
-      retval = 0;
-    }
-
- out_close:
-  if (sock != -1)
-    __close_nocancel_nostatus (sock);
- out:
-  if (__nscd_map_ref_retry_or_drop (&map, &gc_cycle, &nretries, retval))
-    goto retry;
-
-  return retval;
+  return -1;
 }