/*
Unix SMB / CIFS implementation .
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
2007-07-09 19:25:36 +00:00
the Free Software Foundation ; either version 3 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 19:22:49 +00:00
struct passwd * tcopy_passwd ( TALLOC_CTX * mem_ctx , const struct passwd * from )
{
2006-02-03 22:19:41 +00:00
struct passwd * ret = TALLOC_P ( mem_ctx , struct passwd ) ;
2006-03-29 22:41:24 +00:00
if ( ! ret ) {
return NULL ;
}
2006-02-03 22:19:41 +00:00
ret - > pw_name = talloc_strdup ( ret , from - > pw_name ) ;
ret - > pw_passwd = talloc_strdup ( ret , from - > pw_passwd ) ;
ret - > pw_uid = from - > pw_uid ;
ret - > pw_gid = from - > pw_gid ;
2006-02-03 22:19:41 +00: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 ) ;
return ret ;
}
2004-11-12 15:01:40 +00:00
# define PWNAMCACHE_SIZE 4
2006-02-03 22:19:41 +00:00
static struct passwd * * pwnam_cache = NULL ;
2004-11-12 15:01:40 +00:00
static void init_pwnam_cache ( void )
{
2006-02-03 22:19:41 +00:00
if ( pwnam_cache ! = NULL )
2004-11-12 15:01:40 +00:00
return ;
2006-02-03 22:19:41 +00:00
pwnam_cache = TALLOC_ZERO_ARRAY ( NULL , struct passwd * ,
PWNAMCACHE_SIZE ) ;
if ( pwnam_cache = = NULL ) {
2007-06-15 21:58:49 +00:00
smb_panic ( " Could not init pwnam_cache " ) ;
2006-02-03 22:19:41 +00:00
}
2004-11-12 15:01:40 +00:00
return ;
}
2005-03-03 16:52:44 +00:00
void flush_pwnam_cache ( void )
{
2006-02-20 17:59:58 +00:00
TALLOC_FREE ( pwnam_cache ) ;
2006-02-03 22:19:41 +00:00
pwnam_cache = NULL ;
2005-03-03 16:52:44 +00:00
init_pwnam_cache ( ) ;
}
2006-02-03 22:19:41 +00:00
struct passwd * getpwnam_alloc ( TALLOC_CTX * mem_ctx , const char * name )
{
2004-11-12 15:01:40 +00:00
int i ;
struct passwd * temp ;
2004-11-12 15:01:40 +00: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 ) ) ;
2007-04-02 12:52:08 +00:00
return tcopy_passwd ( mem_ctx , pwnam_cache [ i ] ) ;
2004-11-12 15:01:40 +00:00
}
}
temp = sys_getpwnam ( name ) ;
if ( ! temp ) {
#if 0
if ( errno = = ENOMEM ) {
/* what now? */
}
# endif
return NULL ;
}
2004-11-12 15:01:40 +00:00
for ( i = 0 ; i < PWNAMCACHE_SIZE ; i + + ) {
if ( pwnam_cache [ i ] = = NULL )
break ;
}
if ( i = = PWNAMCACHE_SIZE )
i = rand ( ) % PWNAMCACHE_SIZE ;
2006-02-03 22:19:41 +00:00
if ( pwnam_cache [ i ] ! = NULL ) {
2007-04-02 03:46:13 +00:00
/* Remove this old cache entry, from the cache. We
* use talloc_unlink here because we want to be very
* clear which referece we are removing */
talloc_unlink ( pwnam_cache , pwnam_cache [ i ] ) ;
2006-02-03 22:19:41 +00:00
}
2004-11-12 15:01:40 +00:00
2006-02-21 19:22:49 +00:00
pwnam_cache [ i ] = tcopy_passwd ( pwnam_cache , temp ) ;
2007-04-02 06:26:16 +00:00
2007-04-02 06:57:37 +00:00
return tcopy_passwd ( mem_ctx , temp ) ;
}
2006-02-03 22:19:41 +00:00
struct passwd * getpwuid_alloc ( TALLOC_CTX * mem_ctx , uid_t uid )
{
struct passwd * temp ;
temp = sys_getpwuid ( uid ) ;
if ( ! temp ) {
#if 0
if ( errno = = ENOMEM ) {
/* what now? */
}
# endif
return NULL ;
}
2006-02-21 19:22:49 +00:00
return tcopy_passwd ( mem_ctx , temp ) ;
}