2002-01-17 11:45:58 +03:00
/*
2002-01-30 09:08:46 +03:00
Unix SMB / CIFS implementation .
2002-01-17 11:45:58 +03:00
Safe versions of getpw * calls
Copyright ( C ) Andrew Bartlett 2002
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation ; either version 2 of the License , or
( at your option ) any later version .
This program 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 General Public License for more details .
You should have received a copy of the GNU General Public License
along with this program ; if not , write to the Free Software
Foundation , Inc . , 675 Mass Ave , Cambridge , MA 0213 9 , USA .
*/
# include "includes.h"
2006-02-21 22:22:49 +03:00
struct passwd * tcopy_passwd ( TALLOC_CTX * mem_ctx , const struct passwd * from )
2002-01-17 11:45:58 +03:00
{
2006-02-04 01:19:41 +03:00
struct passwd * ret = TALLOC_P ( mem_ctx , struct passwd ) ;
2006-03-30 02:41:24 +04:00
if ( ! ret ) {
return NULL ;
}
2006-02-04 01:19:41 +03:00
ret - > pw_name = talloc_strdup ( ret , from - > pw_name ) ;
ret - > pw_passwd = talloc_strdup ( ret , from - > pw_passwd ) ;
2002-01-17 11:45:58 +03:00
ret - > pw_uid = from - > pw_uid ;
ret - > pw_gid = from - > pw_gid ;
2006-02-04 01:19:41 +03:00
ret - > pw_gecos = talloc_strdup ( ret , from - > pw_gecos ) ;
ret - > pw_dir = talloc_strdup ( ret , from - > pw_dir ) ;
ret - > pw_shell = talloc_strdup ( ret , from - > pw_shell ) ;
2002-01-17 11:45:58 +03:00
return ret ;
}
2004-11-12 18:01:40 +03:00
# define PWNAMCACHE_SIZE 4
2006-02-04 01:19:41 +03:00
static struct passwd * * pwnam_cache = NULL ;
2004-11-12 18:01:40 +03:00
static void init_pwnam_cache ( void )
{
2006-02-04 01:19:41 +03:00
if ( pwnam_cache ! = NULL )
2004-11-12 18:01:40 +03:00
return ;
2006-02-04 01:19:41 +03:00
pwnam_cache = TALLOC_ZERO_ARRAY ( NULL , struct passwd * ,
PWNAMCACHE_SIZE ) ;
if ( pwnam_cache = = NULL ) {
smb_panic ( " Could not init pwnam_cache \n " ) ;
}
2004-11-12 18:01:40 +03:00
return ;
}
2005-03-03 19:52:44 +03:00
void flush_pwnam_cache ( void )
{
2006-02-20 20:59:58 +03:00
TALLOC_FREE ( pwnam_cache ) ;
2006-02-04 01:19:41 +03:00
pwnam_cache = NULL ;
2005-03-03 19:52:44 +03:00
init_pwnam_cache ( ) ;
}
2006-02-04 01:19:41 +03:00
struct passwd * getpwnam_alloc ( TALLOC_CTX * mem_ctx , const char * name )
2002-01-17 11:45:58 +03:00
{
2004-11-12 18:01:40 +03:00
int i ;
2002-01-17 11:45:58 +03:00
struct passwd * temp ;
2004-11-12 18:01:40 +03:00
init_pwnam_cache ( ) ;
for ( i = 0 ; i < PWNAMCACHE_SIZE ; i + + ) {
if ( ( pwnam_cache [ i ] ! = NULL ) & &
( strcmp ( name , pwnam_cache [ i ] - > pw_name ) = = 0 ) ) {
DEBUG ( 10 , ( " Got %s from pwnam_cache \n " , name ) ) ;
2006-07-31 08:30:55 +04:00
return ( struct passwd * ) talloc_reference ( mem_ctx , pwnam_cache [ i ] ) ;
2004-11-12 18:01:40 +03:00
}
}
2002-07-15 14:35:28 +04:00
temp = sys_getpwnam ( name ) ;
2002-01-17 11:45:58 +03:00
if ( ! temp ) {
#if 0
if ( errno = = ENOMEM ) {
/* what now? */
}
# endif
return NULL ;
}
2004-11-12 18:01:40 +03:00
for ( i = 0 ; i < PWNAMCACHE_SIZE ; i + + ) {
if ( pwnam_cache [ i ] = = NULL )
break ;
}
if ( i = = PWNAMCACHE_SIZE )
i = rand ( ) % PWNAMCACHE_SIZE ;
2006-02-04 01:19:41 +03:00
if ( pwnam_cache [ i ] ! = NULL ) {
2006-02-20 20:59:58 +03:00
TALLOC_FREE ( pwnam_cache [ i ] ) ;
2006-02-04 01:19:41 +03:00
}
2004-11-12 18:01:40 +03:00
2006-02-21 22:22:49 +03:00
pwnam_cache [ i ] = tcopy_passwd ( pwnam_cache , temp ) ;
2006-03-30 02:41:24 +04:00
if ( pwnam_cache [ i ] ! = NULL & & mem_ctx ! = NULL ) {
2006-07-31 08:30:55 +04:00
return ( struct passwd * ) talloc_reference ( mem_ctx , pwnam_cache [ i ] ) ;
2006-02-04 01:19:41 +03:00
}
2004-11-12 18:01:40 +03:00
2006-02-21 22:22:49 +03:00
return tcopy_passwd ( NULL , pwnam_cache [ i ] ) ;
2002-01-17 11:45:58 +03:00
}
2006-02-04 01:19:41 +03:00
struct passwd * getpwuid_alloc ( TALLOC_CTX * mem_ctx , uid_t uid )
2002-01-17 11:45:58 +03:00
{
struct passwd * temp ;
2002-07-15 14:35:28 +04:00
temp = sys_getpwuid ( uid ) ;
2002-01-17 11:45:58 +03:00
if ( ! temp ) {
#if 0
if ( errno = = ENOMEM ) {
/* what now? */
}
# endif
return NULL ;
}
2006-02-21 22:22:49 +03:00
return tcopy_passwd ( mem_ctx , temp ) ;
2002-01-17 11:45:58 +03:00
}