2008-12-16 09:30:16 +01:00
/*
2002-01-30 06:08:46 +00:00
Unix SMB / CIFS implementation .
2000-05-09 11:43:00 +00:00
Windows NT Domain nsswitch module
Copyright ( C ) Tim Potter 2000
2008-12-16 09:30:16 +01:00
2000-05-09 11:43:00 +00:00
This library is free software ; you can redistribute it and / or
2007-07-10 04:04:46 +00:00
modify it under the terms of the GNU Lesser General Public
2000-05-09 11:43:00 +00:00
License as published by the Free Software Foundation ; either
2007-07-10 02:31:50 +00:00
version 3 of the License , or ( at your option ) any later version .
2008-12-16 09:30:16 +01:00
2000-05-09 11:43:00 +00:00
This 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
Library General Public License for more details .
2008-12-16 09:30:16 +01:00
2007-07-10 04:04:46 +00:00
You should have received a copy of the GNU Lesser General Public License
2007-07-10 02:31:50 +00:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
2000-05-09 11:43:00 +00:00
*/
2002-09-07 05:41:23 +00:00
# include "winbind_client.h"
2000-05-09 11:43:00 +00:00
2018-11-20 14:08:31 +01:00
# ifdef HAVE_PTHREAD_H
2007-09-28 18:12:22 +00:00
# include <pthread.h>
# endif
2003-03-31 04:36:01 +00:00
/* Maximum number of users to pass back over the unix domain socket
2008-12-16 09:30:16 +01:00
per call . This is not a static limit on the total number of users
2003-03-31 04:36:01 +00:00
or groups returned in total . */
2001-12-22 00:51:32 +00:00
# define MAX_GETPWENT_USERS 250
# define MAX_GETGRENT_USERS 250
2010-01-23 18:20:36 +01:00
/*************************************************************************
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
# ifdef DEBUG_NSS
static const char * nss_err_str ( NSS_STATUS ret )
{
switch ( ret ) {
case NSS_STATUS_TRYAGAIN :
return " NSS_STATUS_TRYAGAIN " ;
case NSS_STATUS_SUCCESS :
return " NSS_STATUS_SUCCESS " ;
case NSS_STATUS_NOTFOUND :
return " NSS_STATUS_NOTFOUND " ;
case NSS_STATUS_UNAVAIL :
return " NSS_STATUS_UNAVAIL " ;
# ifdef NSS_STATUS_RETURN
case NSS_STATUS_RETURN :
return " NSS_STATUS_RETURN " ;
# endif
default :
return " UNKNOWN RETURN CODE!!!!!!! " ;
}
}
# endif
2001-12-22 00:51:32 +00:00
/* Prototypes from wb_common.c */
2000-05-09 11:43:00 +00:00
/* Allocate some space from the nss static buffer. The buffer and buflen
are the pointers passed in by the C library to the _nss_ntdom_ *
functions . */
2004-02-02 21:14:55 +00:00
static char * get_static ( char * * buffer , size_t * buflen , size_t len )
2000-05-09 11:43:00 +00:00
{
2001-04-25 05:47:50 +00:00
char * result ;
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
/* Error check. We return false if things aren't set up right, or
there isn ' t enough buffer space left . */
2007-12-07 17:32:32 -08:00
2001-04-25 05:47:50 +00:00
if ( ( buffer = = NULL ) | | ( buflen = = NULL ) | | ( * buflen < len ) ) {
return NULL ;
}
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
/* Return an index into the static buffer */
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
result = * buffer ;
* buffer + = len ;
* buflen - = len ;
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
return result ;
2000-05-09 11:43:00 +00:00
}
2007-12-07 17:32:32 -08:00
/* I've copied the strtok() replacement function next_token_Xalloc() from
2000-05-09 11:43:00 +00:00
lib / util_str . c as I really don ' t want to have to link in any other
objects if I can possibly avoid it . */
2007-12-07 17:32:32 -08:00
static bool next_token_alloc ( const char * * ptr ,
char * * pp_buff ,
const char * sep )
2000-05-09 11:43:00 +00:00
{
2008-12-17 00:30:21 +01:00
const char * s ;
const char * saved_s ;
2007-12-07 17:32:32 -08:00
char * pbuf ;
2007-09-13 14:14:02 +00:00
bool quoted ;
2001-04-25 05:47:50 +00:00
size_t len = 1 ;
2001-06-21 09:10:42 +00:00
2007-12-07 17:32:32 -08:00
* pp_buff = NULL ;
if ( ! ptr ) {
return ( false ) ;
}
2001-06-21 09:10:42 +00:00
2008-12-17 00:30:21 +01:00
s = * ptr ;
2001-06-21 09:10:42 +00:00
2001-04-25 05:47:50 +00:00
/* default to simple separators */
2007-12-07 17:32:32 -08:00
if ( ! sep ) {
sep = " \t \n \r " ;
}
2001-06-21 09:10:42 +00:00
2001-04-25 05:47:50 +00:00
/* find the first non sep char */
2007-12-07 17:32:32 -08:00
while ( * s & & strchr ( sep , * s ) ) {
s + + ;
}
2001-04-25 05:47:50 +00:00
/* nothing left? */
2007-12-07 17:32:32 -08:00
if ( ! * s ) {
return false ;
}
/* When restarting we need to go from here. */
saved_s = s ;
/* Work out the length needed. */
for ( quoted = false ; * s & &
( quoted | | ! strchr ( sep , * s ) ) ; s + + ) {
2001-04-25 05:47:50 +00:00
if ( * s = = ' \" ' ) {
quoted = ! quoted ;
} else {
len + + ;
}
}
2007-12-07 17:32:32 -08:00
/* We started with len = 1 so we have space for the nul. */
2008-04-06 12:26:15 +02:00
* pp_buff = ( char * ) malloc ( len ) ;
2007-12-07 17:32:32 -08:00
if ( ! * pp_buff ) {
return false ;
}
/* copy over the token */
pbuf = * pp_buff ;
s = saved_s ;
for ( quoted = false ; * s & &
( quoted | | ! strchr ( sep , * s ) ) ; s + + ) {
if ( * s = = ' \" ' ) {
quoted = ! quoted ;
} else {
* pbuf + + = * s ;
}
}
* ptr = ( * s ) ? s + 1 : s ;
* pbuf = 0 ;
2007-09-13 14:14:02 +00:00
return true ;
2000-05-09 11:43:00 +00:00
}
/* Fill a pwent structure from a winbindd_response structure. We use
the static data passed to us by libc to put strings and stuff in .
2001-04-25 05:47:50 +00:00
Return NSS_STATUS_TRYAGAIN if we run out of memory . */
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
static NSS_STATUS fill_pwent ( struct passwd * result ,
struct winbindd_pw * pw ,
2002-07-15 10:54:35 +00:00
char * * buffer , size_t * buflen )
2000-05-09 11:43:00 +00:00
{
2016-03-16 09:37:42 -07:00
size_t len ;
2001-04-25 05:47:50 +00:00
/* User name */
2016-03-16 09:37:42 -07:00
len = strlen ( pw - > pw_name ) + 1 ;
2000-05-09 11:43:00 +00:00
2008-12-16 09:30:16 +01:00
if ( ( result - > pw_name =
2016-03-16 09:37:42 -07:00
get_static ( buffer , buflen , len ) ) = = NULL ) {
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
/* Out of memory */
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
return NSS_STATUS_TRYAGAIN ;
}
2000-05-09 11:43:00 +00:00
2016-03-16 09:37:42 -07:00
memcpy ( result - > pw_name , pw - > pw_name , len ) ;
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
/* Password */
2016-03-16 09:37:42 -07:00
len = strlen ( pw - > pw_passwd ) + 1 ;
2000-05-09 11:43:00 +00:00
2008-12-16 09:30:16 +01:00
if ( ( result - > pw_passwd =
2016-03-16 09:37:42 -07:00
get_static ( buffer , buflen , len ) ) = = NULL ) {
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
/* Out of memory */
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
return NSS_STATUS_TRYAGAIN ;
}
2000-05-09 11:43:00 +00:00
2016-03-16 09:37:42 -07:00
memcpy ( result - > pw_passwd , pw - > pw_passwd , len ) ;
2008-12-16 09:30:16 +01:00
2001-04-25 05:47:50 +00:00
/* [ug]id */
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
result - > pw_uid = pw - > pw_uid ;
result - > pw_gid = pw - > pw_gid ;
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
/* GECOS */
2016-03-16 09:37:42 -07:00
len = strlen ( pw - > pw_gecos ) + 1 ;
2000-05-09 11:43:00 +00:00
2008-12-16 09:30:16 +01:00
if ( ( result - > pw_gecos =
2016-03-16 09:37:42 -07:00
get_static ( buffer , buflen , len ) ) = = NULL ) {
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
/* Out of memory */
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
return NSS_STATUS_TRYAGAIN ;
}
2000-05-09 11:43:00 +00:00
2016-03-16 09:37:42 -07:00
memcpy ( result - > pw_gecos , pw - > pw_gecos , len ) ;
2008-12-16 09:30:16 +01:00
2001-04-25 05:47:50 +00:00
/* Home directory */
2016-03-16 09:37:42 -07:00
len = strlen ( pw - > pw_dir ) + 1 ;
2008-12-16 09:30:16 +01:00
if ( ( result - > pw_dir =
2016-03-16 09:37:42 -07:00
get_static ( buffer , buflen , len ) ) = = NULL ) {
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
/* Out of memory */
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
return NSS_STATUS_TRYAGAIN ;
}
2000-05-09 11:43:00 +00:00
2016-03-16 09:37:42 -07:00
memcpy ( result - > pw_dir , pw - > pw_dir , len ) ;
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
/* Logon shell */
2016-03-16 09:37:42 -07:00
len = strlen ( pw - > pw_shell ) + 1 ;
2008-12-16 09:30:16 +01:00
if ( ( result - > pw_shell =
2016-03-16 09:37:42 -07:00
get_static ( buffer , buflen , len ) ) = = NULL ) {
2008-12-16 09:30:16 +01:00
2001-04-25 05:47:50 +00:00
/* Out of memory */
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
return NSS_STATUS_TRYAGAIN ;
}
2000-05-09 11:43:00 +00:00
2016-03-16 09:37:42 -07:00
memcpy ( result - > pw_shell , pw - > pw_shell , len ) ;
2000-05-09 11:43:00 +00:00
2002-05-10 03:02:22 +00:00
/* The struct passwd for Solaris has some extra fields which must
be initialised or nscd crashes . */
2018-11-20 14:08:31 +01:00
# ifdef HAVE_PASSWD_PW_COMMENT
2002-05-10 03:02:22 +00:00
result - > pw_comment = " " ;
# endif
2018-11-20 14:08:31 +01:00
# ifdef HAVE_PASSWD_PW_AGE
2002-05-10 03:02:22 +00:00
result - > pw_age = " " ;
# endif
2001-04-25 05:47:50 +00:00
return NSS_STATUS_SUCCESS ;
2000-05-09 11:43:00 +00:00
}
/* Fill a grent structure from a winbindd_response structure. We use
the static data passed to us by libc to put strings and stuff in .
2001-04-25 05:47:50 +00:00
Return NSS_STATUS_TRYAGAIN if we run out of memory . */
2000-05-09 11:43:00 +00:00
2002-07-15 10:54:35 +00:00
static NSS_STATUS fill_grent ( struct group * result , struct winbindd_gr * gr ,
2014-02-27 09:38:39 +01:00
const char * gr_mem , char * * buffer , size_t * buflen )
2000-05-09 11:43:00 +00:00
{
2007-12-07 17:32:32 -08:00
char * name ;
2001-04-25 05:47:50 +00:00
int i ;
2001-12-22 00:51:32 +00:00
char * tst ;
2016-03-16 09:37:42 -07:00
size_t len ;
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
/* Group name */
2016-03-16 09:37:42 -07:00
len = strlen ( gr - > gr_name ) + 1 ;
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
if ( ( result - > gr_name =
2016-03-16 09:37:42 -07:00
get_static ( buffer , buflen , len ) ) = = NULL ) {
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
/* Out of memory */
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
return NSS_STATUS_TRYAGAIN ;
}
2000-05-09 11:43:00 +00:00
2016-03-16 09:37:42 -07:00
memcpy ( result - > gr_name , gr - > gr_name , len ) ;
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
/* Password */
2016-03-16 09:37:42 -07:00
len = strlen ( gr - > gr_passwd ) + 1 ;
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
if ( ( result - > gr_passwd =
2016-03-16 09:37:42 -07:00
get_static ( buffer , buflen , len ) ) = = NULL ) {
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
/* Out of memory */
return NSS_STATUS_TRYAGAIN ;
}
2000-05-09 11:43:00 +00:00
2016-03-16 09:37:42 -07:00
memcpy ( result - > gr_passwd , gr - > gr_passwd , len ) ;
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
/* gid */
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
result - > gr_gid = gr - > gr_gid ;
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
/* Group membership */
2000-05-09 11:43:00 +00:00
2012-02-25 21:20:21 +01:00
if ( ! gr_mem ) {
2001-04-25 05:47:50 +00:00
gr - > num_gr_mem = 0 ;
}
2000-05-09 11:43:00 +00:00
2001-12-22 00:51:32 +00:00
/* this next value is a pointer to a pointer so let's align it */
/* Calculate number of extra bytes needed to align on pointer size boundry */
2002-07-15 10:54:35 +00:00
if ( ( i = ( unsigned long ) ( * buffer ) % sizeof ( char * ) ) ! = 0 )
2001-12-22 00:51:32 +00:00
i = sizeof ( char * ) - i ;
2007-12-07 17:32:32 -08:00
if ( ( tst = get_static ( buffer , buflen , ( ( gr - > num_gr_mem + 1 ) *
2001-12-22 00:51:32 +00:00
sizeof ( char * ) + i ) ) ) = = NULL ) {
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
/* Out of memory */
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
return NSS_STATUS_TRYAGAIN ;
}
2001-12-22 00:51:32 +00:00
result - > gr_mem = ( char * * ) ( tst + i ) ;
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
if ( gr - > num_gr_mem = = 0 ) {
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
/* Group is empty */
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
* ( result - > gr_mem ) = NULL ;
return NSS_STATUS_SUCCESS ;
}
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
/* Start looking at extra data */
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
i = 0 ;
2000-05-09 11:43:00 +00:00
2007-12-07 17:32:32 -08:00
while ( next_token_alloc ( ( const char * * ) & gr_mem , & name , " , " ) ) {
2001-04-25 05:47:50 +00:00
/* Allocate space for member */
2016-03-16 09:37:42 -07:00
len = strlen ( name ) + 1 ;
2007-12-07 17:32:32 -08:00
if ( ( ( result - > gr_mem ) [ i ] =
2016-03-16 09:37:42 -07:00
get_static ( buffer , buflen , len ) ) = = NULL ) {
2007-12-07 17:32:32 -08:00
free ( name ) ;
2001-04-25 05:47:50 +00:00
/* Out of memory */
return NSS_STATUS_TRYAGAIN ;
2007-12-07 17:32:32 -08:00
}
2016-03-16 09:37:42 -07:00
memcpy ( ( result - > gr_mem ) [ i ] , name , len ) ;
2007-12-07 17:32:32 -08:00
free ( name ) ;
2001-04-25 05:47:50 +00:00
i + + ;
}
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
/* Terminate list */
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
( result - > gr_mem ) [ i ] = NULL ;
return NSS_STATUS_SUCCESS ;
2000-05-09 11:43:00 +00:00
}
/*
* NSS user functions
*/
2022-11-06 16:57:27 +01:00
static __thread struct winbindd_response getpwent_response ;
2001-04-25 05:47:50 +00:00
2022-11-06 16:57:27 +01:00
static __thread int ndx_pw_cache ; /* Current index into pwd cache */
static __thread int num_pw_cache ; /* Current size of pwd cache */
2001-04-25 05:47:50 +00:00
2000-05-09 11:43:00 +00:00
/* Rewind "file pointer" to start of ntdom password database */
2021-07-01 12:08:16 +02:00
_PUBLIC_ON_LINUX_
2001-04-25 05:47:50 +00:00
NSS_STATUS
2000-05-10 14:48:33 +00:00
_nss_winbind_setpwent ( void )
2000-05-09 11:43:00 +00:00
{
2007-02-15 11:44:06 +00:00
NSS_STATUS ret ;
2001-04-25 05:47:50 +00:00
# ifdef DEBUG_NSS
fprintf ( stderr , " [%5d]: setpwent \n " , getpid ( ) ) ;
# endif
if ( num_pw_cache > 0 ) {
ndx_pw_cache = num_pw_cache = 0 ;
2007-09-14 07:07:59 +00:00
winbindd_free_response ( & getpwent_response ) ;
2001-04-25 05:47:50 +00:00
}
2018-11-05 08:14:48 +01:00
winbind_set_client_name ( " nss_winbind " ) ;
2015-01-23 23:58:53 +00:00
ret = winbindd_request_response ( NULL , WINBINDD_SETPWENT , NULL , NULL ) ;
2007-02-15 11:44:06 +00:00
# ifdef DEBUG_NSS
fprintf ( stderr , " [%5d]: setpwent returns %s (%d) \n " , getpid ( ) ,
nss_err_str ( ret ) , ret ) ;
# endif
2007-09-28 18:12:22 +00:00
2007-02-15 11:44:06 +00:00
return ret ;
2000-05-09 11:43:00 +00:00
}
/* Close ntdom password database "file pointer" */
2021-07-01 12:08:16 +02:00
_PUBLIC_ON_LINUX_
2001-04-25 05:47:50 +00:00
NSS_STATUS
2000-05-10 14:48:33 +00:00
_nss_winbind_endpwent ( void )
2000-05-09 11:43:00 +00:00
{
2007-02-15 11:44:06 +00:00
NSS_STATUS ret ;
2001-04-25 05:47:50 +00:00
# ifdef DEBUG_NSS
fprintf ( stderr , " [%5d]: endpwent \n " , getpid ( ) ) ;
# endif
if ( num_pw_cache > 0 ) {
ndx_pw_cache = num_pw_cache = 0 ;
2007-09-14 07:07:59 +00:00
winbindd_free_response ( & getpwent_response ) ;
2001-04-25 05:47:50 +00:00
}
2018-11-05 08:14:48 +01:00
winbind_set_client_name ( " nss_winbind " ) ;
2015-01-23 23:58:53 +00:00
ret = winbindd_request_response ( NULL , WINBINDD_ENDPWENT , NULL , NULL ) ;
2007-02-15 11:44:06 +00:00
# ifdef DEBUG_NSS
fprintf ( stderr , " [%5d]: endpwent returns %s (%d) \n " , getpid ( ) ,
nss_err_str ( ret ) , ret ) ;
# endif
2007-09-28 18:12:22 +00:00
2007-02-15 11:44:06 +00:00
return ret ;
2000-05-09 11:43:00 +00:00
}
/* Fetch the next password entry from ntdom password database */
2021-07-01 12:08:16 +02:00
_PUBLIC_ON_LINUX_
2001-04-25 05:47:50 +00:00
NSS_STATUS
2008-12-16 09:30:16 +01:00
_nss_winbind_getpwent_r ( struct passwd * result , char * buffer ,
2001-04-25 05:47:50 +00:00
size_t buflen , int * errnop )
2000-05-09 11:43:00 +00:00
{
2001-04-25 05:47:50 +00:00
NSS_STATUS ret ;
struct winbindd_request request ;
2022-11-06 16:57:27 +01:00
static __thread int called_again ;
2001-04-25 05:47:50 +00:00
# ifdef DEBUG_NSS
fprintf ( stderr , " [%5d]: getpwent \n " , getpid ( ) ) ;
# endif
/* Return an entry from the cache if we have one, or if we are
called again because we exceeded our static buffer . */
if ( ( ndx_pw_cache < num_pw_cache ) | | called_again ) {
goto return_result ;
}
/* Else call winbindd to get a bunch of entries */
2008-12-16 09:30:16 +01:00
2001-04-25 05:47:50 +00:00
if ( num_pw_cache > 0 ) {
2007-09-14 07:07:59 +00:00
winbindd_free_response ( & getpwent_response ) ;
2001-04-25 05:47:50 +00:00
}
ZERO_STRUCT ( request ) ;
ZERO_STRUCT ( getpwent_response ) ;
request . data . num_entries = MAX_GETPWENT_USERS ;
2018-11-05 08:14:48 +01:00
winbind_set_client_name ( " nss_winbind " ) ;
2015-01-23 23:58:53 +00:00
ret = winbindd_request_response ( NULL , WINBINDD_GETPWENT , & request ,
2001-04-25 05:47:50 +00:00
& getpwent_response ) ;
if ( ret = = NSS_STATUS_SUCCESS ) {
struct winbindd_pw * pw_cache ;
/* Fill cache */
ndx_pw_cache = 0 ;
num_pw_cache = getpwent_response . data . num_entries ;
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
/* Return a result */
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
return_result :
2006-08-01 12:45:12 +00:00
pw_cache = ( struct winbindd_pw * )
getpwent_response . extra_data . data ;
2001-04-25 05:47:50 +00:00
/* Check data is valid */
if ( pw_cache = = NULL ) {
2007-02-15 11:44:06 +00:00
ret = NSS_STATUS_NOTFOUND ;
goto done ;
2001-04-25 05:47:50 +00:00
}
ret = fill_pwent ( result , & pw_cache [ ndx_pw_cache ] ,
& buffer , & buflen ) ;
2008-12-16 09:30:16 +01:00
2001-04-25 05:47:50 +00:00
/* Out of memory - try again */
if ( ret = = NSS_STATUS_TRYAGAIN ) {
2007-09-13 14:14:02 +00:00
called_again = true ;
2001-04-25 05:47:50 +00:00
* errnop = errno = ERANGE ;
2007-02-15 11:44:06 +00:00
goto done ;
2001-04-25 05:47:50 +00:00
}
* errnop = errno = 0 ;
2007-09-13 14:14:02 +00:00
called_again = false ;
2001-04-25 05:47:50 +00:00
ndx_pw_cache + + ;
/* If we've finished with this lot of results free cache */
if ( ndx_pw_cache = = num_pw_cache ) {
ndx_pw_cache = num_pw_cache = 0 ;
2007-09-14 07:07:59 +00:00
winbindd_free_response ( & getpwent_response ) ;
2001-04-25 05:47:50 +00:00
}
}
2007-02-15 11:44:06 +00:00
done :
# ifdef DEBUG_NSS
fprintf ( stderr , " [%5d]: getpwent returns %s (%d) \n " , getpid ( ) ,
nss_err_str ( ret ) , ret ) ;
# endif
2007-09-28 18:12:22 +00:00
2001-04-25 05:47:50 +00:00
return ret ;
2000-05-09 11:43:00 +00:00
}
/* Return passwd struct from uid */
2021-07-01 12:08:16 +02:00
_PUBLIC_ON_LINUX_
2001-04-25 05:47:50 +00:00
NSS_STATUS
2000-05-10 14:48:33 +00:00
_nss_winbind_getpwuid_r ( uid_t uid , struct passwd * result , char * buffer ,
2001-04-25 05:47:50 +00:00
size_t buflen , int * errnop )
2000-05-09 11:43:00 +00:00
{
2001-04-25 05:47:50 +00:00
NSS_STATUS ret ;
2022-11-06 16:57:27 +01:00
static __thread struct winbindd_response response ;
2000-05-09 11:43:00 +00:00
struct winbindd_request request ;
2022-11-06 16:57:27 +01:00
static __thread int keep_response ;
2001-04-25 05:47:50 +00:00
2007-02-15 11:44:06 +00:00
# ifdef DEBUG_NSS
2007-09-28 18:12:22 +00:00
fprintf ( stderr , " [%5d]: getpwuid_r %d \n " , getpid ( ) , ( unsigned int ) uid ) ;
# endif
2001-04-25 05:47:50 +00:00
/* If our static buffer needs to be expanded we are called again */
2007-09-28 18:12:22 +00:00
if ( ! keep_response | | uid ! = response . data . pw . pw_uid ) {
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
/* Call for the first time */
2000-05-09 11:43:00 +00:00
2018-12-21 05:54:31 -06:00
response = ( struct winbindd_response ) {
. length = 0 ,
} ;
request = ( struct winbindd_request ) {
. wb_flags = WBFLAG_FROM_NSS ,
. data = {
. uid = uid ,
} ,
} ;
2001-04-25 05:47:50 +00:00
2018-11-05 08:14:48 +01:00
winbind_set_client_name ( " nss_winbind " ) ;
2015-01-23 23:58:53 +00:00
ret = winbindd_request_response ( NULL , WINBINDD_GETPWUID , & request , & response ) ;
2001-04-25 05:47:50 +00:00
if ( ret = = NSS_STATUS_SUCCESS ) {
2008-12-16 09:30:16 +01:00
ret = fill_pwent ( result , & response . data . pw ,
2001-04-25 05:47:50 +00:00
& buffer , & buflen ) ;
if ( ret = = NSS_STATUS_TRYAGAIN ) {
2007-09-13 14:14:02 +00:00
keep_response = true ;
2001-04-25 05:47:50 +00:00
* errnop = errno = ERANGE ;
2007-02-15 11:44:06 +00:00
goto done ;
2001-04-25 05:47:50 +00:00
}
}
} else {
/* We've been called again */
ret = fill_pwent ( result , & response . data . pw , & buffer , & buflen ) ;
if ( ret = = NSS_STATUS_TRYAGAIN ) {
* errnop = errno = ERANGE ;
2007-02-15 11:44:06 +00:00
goto done ;
2001-04-25 05:47:50 +00:00
}
2007-09-13 14:14:02 +00:00
keep_response = false ;
2001-04-25 05:47:50 +00:00
* errnop = errno = 0 ;
}
2007-09-14 07:07:59 +00:00
winbindd_free_response ( & response ) ;
2007-09-28 18:12:22 +00:00
2007-02-15 11:44:06 +00:00
done :
# ifdef DEBUG_NSS
fprintf ( stderr , " [%5d]: getpwuid %d returns %s (%d) \n " , getpid ( ) ,
( unsigned int ) uid , nss_err_str ( ret ) , ret ) ;
# endif
2007-09-28 18:12:22 +00:00
2001-04-25 05:47:50 +00:00
return ret ;
2000-05-09 11:43:00 +00:00
}
/* Return passwd struct from username */
2021-07-01 12:08:16 +02:00
_PUBLIC_ON_LINUX_
2001-04-25 05:47:50 +00:00
NSS_STATUS
2000-05-10 14:48:33 +00:00
_nss_winbind_getpwnam_r ( const char * name , struct passwd * result , char * buffer ,
2001-04-25 05:47:50 +00:00
size_t buflen , int * errnop )
2000-05-09 11:43:00 +00:00
{
2001-04-25 05:47:50 +00:00
NSS_STATUS ret ;
2022-11-06 16:57:27 +01:00
static __thread struct winbindd_response response ;
2000-05-09 11:43:00 +00:00
struct winbindd_request request ;
2022-11-06 16:57:27 +01:00
static __thread int keep_response ;
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
# ifdef DEBUG_NSS
2007-09-28 18:12:22 +00:00
fprintf ( stderr , " [%5d]: getpwnam_r %s \n " , getpid ( ) , name ) ;
# endif
2001-04-25 05:47:50 +00:00
/* If our static buffer needs to be expanded we are called again */
2000-05-09 11:43:00 +00:00
2007-09-28 18:12:22 +00:00
if ( ! keep_response | | strcmp ( name , response . data . pw . pw_name ) ! = 0 ) {
2001-04-25 05:47:50 +00:00
/* Call for the first time */
2018-12-21 05:54:31 -06:00
response = ( struct winbindd_response ) {
. length = 0 ,
} ;
request = ( struct winbindd_request ) {
. wb_flags = WBFLAG_FROM_NSS ,
} ;
2001-04-25 05:47:50 +00:00
2008-12-16 09:30:16 +01:00
strncpy ( request . data . username , name ,
2001-04-25 05:47:50 +00:00
sizeof ( request . data . username ) - 1 ) ;
request . data . username
[ sizeof ( request . data . username ) - 1 ] = ' \0 ' ;
2018-11-05 08:14:48 +01:00
winbind_set_client_name ( " nss_winbind " ) ;
2015-01-23 23:58:53 +00:00
ret = winbindd_request_response ( NULL , WINBINDD_GETPWNAM , & request , & response ) ;
2001-04-25 05:47:50 +00:00
if ( ret = = NSS_STATUS_SUCCESS ) {
ret = fill_pwent ( result , & response . data . pw , & buffer ,
& buflen ) ;
if ( ret = = NSS_STATUS_TRYAGAIN ) {
2007-09-13 14:14:02 +00:00
keep_response = true ;
2001-04-25 05:47:50 +00:00
* errnop = errno = ERANGE ;
2007-02-15 11:44:06 +00:00
goto done ;
2001-04-25 05:47:50 +00:00
}
}
} else {
/* We've been called again */
ret = fill_pwent ( result , & response . data . pw , & buffer , & buflen ) ;
if ( ret = = NSS_STATUS_TRYAGAIN ) {
2007-09-13 14:14:02 +00:00
keep_response = true ;
2001-04-25 05:47:50 +00:00
* errnop = errno = ERANGE ;
2007-02-15 11:44:06 +00:00
goto done ;
2001-04-25 05:47:50 +00:00
}
2007-09-13 14:14:02 +00:00
keep_response = false ;
2001-04-25 05:47:50 +00:00
* errnop = errno = 0 ;
}
2007-09-14 07:07:59 +00:00
winbindd_free_response ( & response ) ;
2007-02-15 11:44:06 +00:00
done :
# ifdef DEBUG_NSS
fprintf ( stderr , " [%5d]: getpwnam %s returns %s (%d) \n " , getpid ( ) ,
name , nss_err_str ( ret ) , ret ) ;
# endif
2007-09-28 18:12:22 +00:00
2001-04-25 05:47:50 +00:00
return ret ;
2000-05-09 11:43:00 +00:00
}
/*
* NSS group functions
*/
2022-11-06 16:57:27 +01:00
static __thread struct winbindd_response getgrent_response ;
2001-04-25 05:47:50 +00:00
2022-11-06 16:57:27 +01:00
static __thread int ndx_gr_cache ; /* Current index into grp cache */
static __thread int num_gr_cache ; /* Current size of grp cache */
2001-04-25 05:47:50 +00:00
2000-05-09 11:43:00 +00:00
/* Rewind "file pointer" to start of ntdom group database */
2021-07-01 12:08:16 +02:00
_PUBLIC_ON_LINUX_
2001-04-25 05:47:50 +00:00
NSS_STATUS
2000-05-10 14:48:33 +00:00
_nss_winbind_setgrent ( void )
2000-05-09 11:43:00 +00:00
{
2007-02-15 11:44:06 +00:00
NSS_STATUS ret ;
2001-04-25 05:47:50 +00:00
# ifdef DEBUG_NSS
fprintf ( stderr , " [%5d]: setgrent \n " , getpid ( ) ) ;
# endif
if ( num_gr_cache > 0 ) {
ndx_gr_cache = num_gr_cache = 0 ;
2007-09-14 07:07:59 +00:00
winbindd_free_response ( & getgrent_response ) ;
2001-04-25 05:47:50 +00:00
}
2018-11-05 08:14:48 +01:00
winbind_set_client_name ( " nss_winbind " ) ;
2015-01-23 23:58:53 +00:00
ret = winbindd_request_response ( NULL , WINBINDD_SETGRENT , NULL , NULL ) ;
2007-02-15 11:44:06 +00:00
# ifdef DEBUG_NSS
fprintf ( stderr , " [%5d]: setgrent returns %s (%d) \n " , getpid ( ) ,
nss_err_str ( ret ) , ret ) ;
# endif
2007-09-28 18:12:22 +00:00
2007-02-15 11:44:06 +00:00
return ret ;
2000-05-09 11:43:00 +00:00
}
/* Close "file pointer" for ntdom group database */
2021-07-01 12:08:16 +02:00
_PUBLIC_ON_LINUX_
2001-04-25 05:47:50 +00:00
NSS_STATUS
2000-05-10 14:48:33 +00:00
_nss_winbind_endgrent ( void )
2000-05-09 11:43:00 +00:00
{
2007-02-15 11:44:06 +00:00
NSS_STATUS ret ;
2001-04-25 05:47:50 +00:00
# ifdef DEBUG_NSS
fprintf ( stderr , " [%5d]: endgrent \n " , getpid ( ) ) ;
# endif
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
if ( num_gr_cache > 0 ) {
ndx_gr_cache = num_gr_cache = 0 ;
2007-09-14 07:07:59 +00:00
winbindd_free_response ( & getgrent_response ) ;
2001-04-25 05:47:50 +00:00
}
2000-05-09 11:43:00 +00:00
2018-11-05 08:14:48 +01:00
winbind_set_client_name ( " nss_winbind " ) ;
2015-01-23 23:58:53 +00:00
ret = winbindd_request_response ( NULL , WINBINDD_ENDGRENT , NULL , NULL ) ;
2007-02-15 11:44:06 +00:00
# ifdef DEBUG_NSS
fprintf ( stderr , " [%5d]: endgrent returns %s (%d) \n " , getpid ( ) ,
nss_err_str ( ret ) , ret ) ;
# endif
2007-09-28 18:12:22 +00:00
2007-02-15 11:44:06 +00:00
return ret ;
2001-04-25 05:47:50 +00:00
}
2000-05-09 11:43:00 +00:00
/* Get next entry from ntdom group database */
2002-07-14 23:45:55 +00:00
static NSS_STATUS
winbind_getgrent ( enum winbindd_cmd cmd ,
struct group * result ,
char * buffer , size_t buflen , int * errnop )
2000-05-09 11:43:00 +00:00
{
2001-04-25 05:47:50 +00:00
NSS_STATUS ret ;
2022-11-06 16:57:27 +01:00
static __thread struct winbindd_request request ;
static __thread int called_again ;
2008-12-16 09:30:16 +01:00
2001-04-25 05:47:50 +00:00
# ifdef DEBUG_NSS
fprintf ( stderr , " [%5d]: getgrent \n " , getpid ( ) ) ;
# endif
/* Return an entry from the cache if we have one, or if we are
called again because we exceeded our static buffer . */
if ( ( ndx_gr_cache < num_gr_cache ) | | called_again ) {
goto return_result ;
}
/* Else call winbindd to get a bunch of entries */
2008-12-16 09:30:16 +01:00
2001-04-25 05:47:50 +00:00
if ( num_gr_cache > 0 ) {
2007-09-14 07:07:59 +00:00
winbindd_free_response ( & getgrent_response ) ;
2001-04-25 05:47:50 +00:00
}
ZERO_STRUCT ( request ) ;
ZERO_STRUCT ( getgrent_response ) ;
request . data . num_entries = MAX_GETGRENT_USERS ;
2018-11-05 08:14:48 +01:00
winbind_set_client_name ( " nss_winbind " ) ;
2015-01-23 23:58:53 +00:00
ret = winbindd_request_response ( NULL , cmd , & request ,
2001-04-25 05:47:50 +00:00
& getgrent_response ) ;
if ( ret = = NSS_STATUS_SUCCESS ) {
struct winbindd_gr * gr_cache ;
int mem_ofs ;
/* Fill cache */
ndx_gr_cache = 0 ;
num_gr_cache = getgrent_response . data . num_entries ;
/* Return a result */
return_result :
2006-08-01 12:45:12 +00:00
gr_cache = ( struct winbindd_gr * )
getgrent_response . extra_data . data ;
2001-04-25 05:47:50 +00:00
/* Check data is valid */
if ( gr_cache = = NULL ) {
2007-02-15 11:44:06 +00:00
ret = NSS_STATUS_NOTFOUND ;
goto done ;
2001-04-25 05:47:50 +00:00
}
/* Fill group membership. The offset into the extra data
for the group membership is the reported offset plus the
size of all the winbindd_gr records returned . */
mem_ofs = gr_cache [ ndx_gr_cache ] . gr_mem_ofs +
num_gr_cache * sizeof ( struct winbindd_gr ) ;
ret = fill_grent ( result , & gr_cache [ ndx_gr_cache ] ,
2006-04-12 14:10:39 +00:00
( ( char * ) getgrent_response . extra_data . data ) + mem_ofs ,
2001-06-05 02:00:47 +00:00
& buffer , & buflen ) ;
2008-12-16 09:30:16 +01:00
2001-04-25 05:47:50 +00:00
/* Out of memory - try again */
if ( ret = = NSS_STATUS_TRYAGAIN ) {
2007-09-13 14:14:02 +00:00
called_again = true ;
2001-04-25 05:47:50 +00:00
* errnop = errno = ERANGE ;
2007-02-15 11:44:06 +00:00
goto done ;
2001-04-25 05:47:50 +00:00
}
* errnop = 0 ;
2007-09-13 14:14:02 +00:00
called_again = false ;
2001-04-25 05:47:50 +00:00
ndx_gr_cache + + ;
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
/* If we've finished with this lot of results free cache */
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
if ( ndx_gr_cache = = num_gr_cache ) {
ndx_gr_cache = num_gr_cache = 0 ;
2007-09-14 07:07:59 +00:00
winbindd_free_response ( & getgrent_response ) ;
2001-04-25 05:47:50 +00:00
}
}
2007-02-15 11:44:06 +00:00
done :
# ifdef DEBUG_NSS
fprintf ( stderr , " [%5d]: getgrent returns %s (%d) \n " , getpid ( ) ,
nss_err_str ( ret ) , ret ) ;
# endif
2007-09-28 18:12:22 +00:00
2001-04-25 05:47:50 +00:00
return ret ;
2000-05-09 11:43:00 +00:00
}
2002-07-14 23:45:55 +00:00
2021-07-01 12:08:16 +02:00
_PUBLIC_ON_LINUX_
2002-07-14 23:45:55 +00:00
NSS_STATUS
_nss_winbind_getgrent_r ( struct group * result ,
char * buffer , size_t buflen , int * errnop )
{
return winbind_getgrent ( WINBINDD_GETGRENT , result , buffer , buflen , errnop ) ;
}
2021-07-01 12:08:16 +02:00
_PUBLIC_ON_LINUX_
2002-07-14 23:45:55 +00:00
NSS_STATUS
_nss_winbind_getgrlst_r ( struct group * result ,
char * buffer , size_t buflen , int * errnop )
{
return winbind_getgrent ( WINBINDD_GETGRLST , result , buffer , buflen , errnop ) ;
}
2000-05-09 11:43:00 +00:00
/* Return group struct from group name */
2021-07-01 12:08:16 +02:00
_PUBLIC_ON_LINUX_
2001-04-25 05:47:50 +00:00
NSS_STATUS
2000-05-10 14:48:33 +00:00
_nss_winbind_getgrnam_r ( const char * name ,
2001-04-25 05:47:50 +00:00
struct group * result , char * buffer ,
size_t buflen , int * errnop )
2000-05-09 11:43:00 +00:00
{
2001-04-25 05:47:50 +00:00
NSS_STATUS ret ;
2022-11-06 16:57:27 +01:00
static __thread struct winbindd_response response ;
2000-05-09 11:43:00 +00:00
struct winbindd_request request ;
2022-11-06 16:57:27 +01:00
static __thread int keep_response ;
2008-12-16 09:30:16 +01:00
2001-04-25 05:47:50 +00:00
# ifdef DEBUG_NSS
fprintf ( stderr , " [%5d]: getgrnam %s \n " , getpid ( ) , name ) ;
# endif
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
/* If our static buffer needs to be expanded we are called again */
2007-09-28 18:12:22 +00:00
/* Or if the stored response group name differs from the request. */
if ( ! keep_response | | strcmp ( name , response . data . gr . gr_name ) ! = 0 ) {
2001-04-25 05:47:50 +00:00
/* Call for the first time */
2018-12-21 05:54:31 -06:00
response = ( struct winbindd_response ) {
. length = 0 ,
} ;
request = ( struct winbindd_request ) {
. wb_flags = WBFLAG_FROM_NSS ,
} ;
2001-04-25 05:47:50 +00:00
2008-12-16 09:30:16 +01:00
strncpy ( request . data . groupname , name ,
2001-04-25 05:47:50 +00:00
sizeof ( request . data . groupname ) ) ;
request . data . groupname
[ sizeof ( request . data . groupname ) - 1 ] = ' \0 ' ;
2018-11-05 08:14:48 +01:00
winbind_set_client_name ( " nss_winbind " ) ;
2015-01-23 23:58:53 +00:00
ret = winbindd_request_response ( NULL , WINBINDD_GETGRNAM ,
& request , & response ) ;
2001-04-25 05:47:50 +00:00
if ( ret = = NSS_STATUS_SUCCESS ) {
2008-12-16 09:30:16 +01:00
ret = fill_grent ( result , & response . data . gr ,
2006-08-01 12:45:12 +00:00
( char * ) response . extra_data . data ,
2001-04-25 05:47:50 +00:00
& buffer , & buflen ) ;
if ( ret = = NSS_STATUS_TRYAGAIN ) {
2007-09-13 14:14:02 +00:00
keep_response = true ;
2001-04-25 05:47:50 +00:00
* errnop = errno = ERANGE ;
2007-02-15 11:44:06 +00:00
goto done ;
2001-04-25 05:47:50 +00:00
}
}
} else {
2007-09-28 18:12:22 +00:00
2001-04-25 05:47:50 +00:00
/* We've been called again */
2007-09-28 18:12:22 +00:00
2008-12-16 09:30:16 +01:00
ret = fill_grent ( result , & response . data . gr ,
2006-08-01 12:45:12 +00:00
( char * ) response . extra_data . data , & buffer ,
& buflen ) ;
2007-09-28 18:12:22 +00:00
2001-04-25 05:47:50 +00:00
if ( ret = = NSS_STATUS_TRYAGAIN ) {
2007-09-13 14:14:02 +00:00
keep_response = true ;
2001-04-25 05:47:50 +00:00
* errnop = errno = ERANGE ;
2007-02-15 11:44:06 +00:00
goto done ;
2001-04-25 05:47:50 +00:00
}
2007-09-13 14:14:02 +00:00
keep_response = false ;
2001-04-25 05:47:50 +00:00
* errnop = 0 ;
}
2000-05-09 11:43:00 +00:00
2007-09-14 07:07:59 +00:00
winbindd_free_response ( & response ) ;
2007-02-15 11:44:06 +00:00
done :
# ifdef DEBUG_NSS
fprintf ( stderr , " [%5d]: getgrnam %s returns %s (%d) \n " , getpid ( ) ,
name , nss_err_str ( ret ) , ret ) ;
# endif
2007-09-28 18:12:22 +00:00
2001-04-25 05:47:50 +00:00
return ret ;
2000-05-09 11:43:00 +00:00
}
/* Return group struct from gid */
2021-07-01 12:08:16 +02:00
_PUBLIC_ON_LINUX_
2001-04-25 05:47:50 +00:00
NSS_STATUS
2000-05-10 14:48:33 +00:00
_nss_winbind_getgrgid_r ( gid_t gid ,
2001-04-25 05:47:50 +00:00
struct group * result , char * buffer ,
size_t buflen , int * errnop )
2000-05-09 11:43:00 +00:00
{
2001-04-25 05:47:50 +00:00
NSS_STATUS ret ;
2022-11-06 16:57:27 +01:00
static __thread struct winbindd_response response ;
2001-04-25 05:47:50 +00:00
struct winbindd_request request ;
2022-11-06 16:57:27 +01:00
static __thread int keep_response ;
2001-04-25 05:47:50 +00:00
# ifdef DEBUG_NSS
fprintf ( stderr , " [%5d]: getgrgid %d \n " , getpid ( ) , gid ) ;
# endif
/* If our static buffer needs to be expanded we are called again */
2007-09-28 18:12:22 +00:00
/* Or if the stored response group name differs from the request. */
2001-04-25 05:47:50 +00:00
2007-09-28 18:12:22 +00:00
if ( ! keep_response | | gid ! = response . data . gr . gr_gid ) {
2001-04-25 05:47:50 +00:00
/* Call for the first time */
2018-12-21 05:54:31 -06:00
response = ( struct winbindd_response ) {
. length = 0 ,
} ;
request = ( struct winbindd_request ) {
. wb_flags = WBFLAG_FROM_NSS ,
} ;
2001-04-25 05:47:50 +00:00
request . data . gid = gid ;
2018-11-05 08:14:48 +01:00
winbind_set_client_name ( " nss_winbind " ) ;
2015-01-23 23:58:53 +00:00
ret = winbindd_request_response ( NULL , WINBINDD_GETGRGID ,
& request , & response ) ;
2001-04-25 05:47:50 +00:00
if ( ret = = NSS_STATUS_SUCCESS ) {
2008-12-16 09:30:16 +01:00
ret = fill_grent ( result , & response . data . gr ,
( char * ) response . extra_data . data ,
2001-04-25 05:47:50 +00:00
& buffer , & buflen ) ;
if ( ret = = NSS_STATUS_TRYAGAIN ) {
2007-09-13 14:14:02 +00:00
keep_response = true ;
2001-04-25 05:47:50 +00:00
* errnop = errno = ERANGE ;
2007-02-15 11:44:06 +00:00
goto done ;
2001-04-25 05:47:50 +00:00
}
}
} else {
/* We've been called again */
2008-12-16 09:30:16 +01:00
ret = fill_grent ( result , & response . data . gr ,
2006-08-01 12:45:12 +00:00
( char * ) response . extra_data . data , & buffer ,
& buflen ) ;
2001-04-25 05:47:50 +00:00
if ( ret = = NSS_STATUS_TRYAGAIN ) {
2007-09-13 14:14:02 +00:00
keep_response = true ;
2001-04-25 05:47:50 +00:00
* errnop = errno = ERANGE ;
2007-02-15 11:44:06 +00:00
goto done ;
2001-04-25 05:47:50 +00:00
}
2007-09-13 14:14:02 +00:00
keep_response = false ;
2001-04-25 05:47:50 +00:00
* errnop = 0 ;
}
2007-09-14 07:07:59 +00:00
winbindd_free_response ( & response ) ;
2007-02-15 11:44:06 +00:00
done :
# ifdef DEBUG_NSS
fprintf ( stderr , " [%5d]: getgrgid %d returns %s (%d) \n " , getpid ( ) ,
( unsigned int ) gid , nss_err_str ( ret ) , ret ) ;
# endif
2007-09-28 18:12:22 +00:00
2001-04-25 05:47:50 +00:00
return ret ;
}
/* Initialise supplementary groups */
2021-07-01 12:08:16 +02:00
_PUBLIC_ON_LINUX_
2001-04-25 05:47:50 +00:00
NSS_STATUS
2017-10-21 14:14:34 +02:00
_nss_winbind_initgroups_dyn ( const char * user , gid_t group , long int * start ,
2001-12-03 03:53:57 +00:00
long int * size , gid_t * * groups , long int limit ,
int * errnop )
2001-04-25 05:47:50 +00:00
{
NSS_STATUS ret ;
2000-05-09 11:43:00 +00:00
struct winbindd_request request ;
2001-04-25 05:47:50 +00:00
struct winbindd_response response ;
int i ;
# ifdef DEBUG_NSS
fprintf ( stderr , " [%5d]: initgroups %s (%d) \n " , getpid ( ) ,
user , group ) ;
# endif
ZERO_STRUCT ( request ) ;
ZERO_STRUCT ( response ) ;
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
strncpy ( request . data . username , user ,
sizeof ( request . data . username ) - 1 ) ;
2000-05-09 11:43:00 +00:00
2018-11-05 08:14:48 +01:00
winbind_set_client_name ( " nss_winbind " ) ;
2015-01-23 23:58:53 +00:00
ret = winbindd_request_response ( NULL , WINBINDD_GETGROUPS ,
& request , & response ) ;
2001-04-25 05:47:50 +00:00
if ( ret = = NSS_STATUS_SUCCESS ) {
int num_gids = response . data . num_entries ;
2006-04-12 14:10:39 +00:00
gid_t * gid_list = ( gid_t * ) response . extra_data . data ;
2001-04-25 05:47:50 +00:00
2007-02-15 12:10:57 +00:00
# ifdef DEBUG_NSS
fprintf ( stderr , " [%5d]: initgroups %s: got NSS_STATUS_SUCCESS "
" and %d gids \n " , getpid ( ) ,
user , num_gids ) ;
# endif
2007-02-16 17:25:54 +00:00
if ( gid_list = = NULL ) {
ret = NSS_STATUS_NOTFOUND ;
goto done ;
}
2007-02-15 12:10:57 +00:00
2001-04-25 05:47:50 +00:00
/* Copy group list to client */
for ( i = 0 ; i < num_gids ; i + + ) {
2007-02-15 12:10:57 +00:00
# ifdef DEBUG_NSS
fprintf ( stderr , " [%5d]: initgroups %s (%d): "
" processing gid %d \n " , getpid ( ) ,
user , group , gid_list [ i ] ) ;
# endif
2001-04-25 05:47:50 +00:00
/* Skip primary group */
2004-09-20 20:18:19 +00:00
if ( gid_list [ i ] = = group ) {
continue ;
2014-09-19 13:33:10 +02:00
}
/* Skip groups without a mapping */
if ( gid_list [ i ] = = ( uid_t ) - 1 ) {
continue ;
2004-09-20 20:18:19 +00:00
}
2001-04-25 05:47:50 +00:00
2004-09-20 20:18:19 +00:00
/* Filled buffer ? If so, resize. */
if ( * start = = * size ) {
long int newsize ;
gid_t * newgroups ;
newsize = 2 * ( * size ) ;
if ( limit > 0 ) {
if ( * size = = limit ) {
goto done ;
}
2004-09-22 23:27:20 +00:00
if ( newsize > limit ) {
newsize = limit ;
}
2004-09-20 20:18:19 +00:00
}
2006-08-01 12:45:12 +00:00
newgroups = ( gid_t * )
realloc ( ( * groups ) ,
newsize * sizeof ( * * groups ) ) ;
2004-09-20 20:18:19 +00:00
if ( ! newgroups ) {
* errnop = ENOMEM ;
ret = NSS_STATUS_NOTFOUND ;
goto done ;
}
* groups = newgroups ;
* size = newsize ;
2001-04-25 05:47:50 +00:00
}
2004-09-20 20:18:19 +00:00
/* Add to buffer */
2001-12-03 03:53:57 +00:00
( * groups ) [ * start ] = gid_list [ i ] ;
2001-04-25 05:47:50 +00:00
* start + = 1 ;
}
}
2008-12-16 09:30:16 +01:00
2001-04-25 05:47:50 +00:00
/* Back to your regularly scheduled programming */
2000-05-09 11:43:00 +00:00
2001-04-25 05:47:50 +00:00
done :
2022-12-09 10:55:53 +00:00
winbindd_free_response ( & response ) ;
2007-02-15 11:44:06 +00:00
# ifdef DEBUG_NSS
fprintf ( stderr , " [%5d]: initgroups %s returns %s (%d) \n " , getpid ( ) ,
user , nss_err_str ( ret ) , ret ) ;
# endif
2007-09-28 18:12:22 +00:00
2001-04-25 05:47:50 +00:00
return ret ;
2000-05-09 11:43:00 +00:00
}