2006-08-19 01:04:54 +00:00
/*
Unix SMB / CIFS implementation .
Winbind daemon - cached credentials funcions
Copyright ( C ) Robert O ' Callahan 2006
Copyright ( C ) Jeremy Allison 2006 ( minor fixes to fit into Samba and
protect against integer wrap ) .
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"
# include "winbindd.h"
# undef DBGC_CLASS
# define DBGC_CLASS DBGC_WINBIND
static BOOL client_can_access_ccache_entry ( uid_t client_uid ,
2006-08-28 23:01:30 +00:00
struct WINBINDD_MEMORY_CREDS * entry )
2006-08-19 01:04:54 +00:00
{
if ( client_uid = = entry - > uid | | client_uid = = 0 ) {
DEBUG ( 10 , ( " Access granted to uid %d \n " , client_uid ) ) ;
return True ;
}
DEBUG ( 1 , ( " Access denied to uid %d (expected %d) \n " , client_uid , entry - > uid ) ) ;
return False ;
}
2006-08-20 01:25:26 +00:00
static NTSTATUS do_ntlm_auth_with_hashes ( const char * username ,
2006-08-19 01:04:54 +00:00
const char * domain ,
2006-08-20 01:25:26 +00:00
const unsigned char lm_hash [ LM_HASH_LEN ] ,
const unsigned char nt_hash [ NT_HASH_LEN ] ,
2006-08-19 01:04:54 +00:00
const DATA_BLOB initial_msg ,
const DATA_BLOB challenge_msg ,
DATA_BLOB * auth_msg )
{
NTSTATUS status ;
NTLMSSP_STATE * ntlmssp_state = NULL ;
DATA_BLOB dummy_msg , reply ;
status = ntlmssp_client_start ( & ntlmssp_state ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 1 , ( " Could not start NTLMSSP client: %s \n " ,
nt_errstr ( status ) ) ) ;
goto done ;
}
status = ntlmssp_set_username ( ntlmssp_state , username ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 1 , ( " Could not set username: %s \n " ,
nt_errstr ( status ) ) ) ;
goto done ;
}
status = ntlmssp_set_domain ( ntlmssp_state , domain ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 1 , ( " Could not set domain: %s \n " ,
nt_errstr ( status ) ) ) ;
goto done ;
}
2006-08-20 01:25:26 +00:00
status = ntlmssp_set_hashes ( ntlmssp_state , lm_hash , nt_hash ) ;
2006-08-19 01:04:54 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2006-08-20 01:25:26 +00:00
DEBUG ( 1 , ( " Could not set hashes: %s \n " ,
2006-08-19 01:04:54 +00:00
nt_errstr ( status ) ) ) ;
goto done ;
}
/* We need to get our protocol handler into the right state. So first
we ask it to generate the initial message . Actually the client has already
sent its own initial message , so we ' re going to drop this one on the floor .
The client might have sent a different message , for example with different
negotiation options , but as far as I can tell this won ' t hurt us . ( Unless
the client sent a different username or domain , in which case that ' s their
problem for telling us the wrong username or domain . )
Since we have a copy of the initial message that the client sent , we could
resolve any discrepancies if we had to .
*/
dummy_msg = data_blob ( NULL , 0 ) ;
reply = data_blob ( NULL , 0 ) ;
status = ntlmssp_update ( ntlmssp_state , dummy_msg , & reply ) ;
data_blob_free ( & dummy_msg ) ;
data_blob_free ( & reply ) ;
if ( ! NT_STATUS_EQUAL ( status , NT_STATUS_MORE_PROCESSING_REQUIRED ) ) {
DEBUG ( 1 , ( " Failed to create initial message! [%s] \n " ,
nt_errstr ( status ) ) ) ;
goto done ;
}
/* Now we are ready to handle the server's actual response. */
status = ntlmssp_update ( ntlmssp_state , challenge_msg , & reply ) ;
if ( ! NT_STATUS_EQUAL ( status , NT_STATUS_OK ) ) {
DEBUG ( 1 , ( " We didn't get a response to the challenge! [%s] \n " ,
nt_errstr ( status ) ) ) ;
data_blob_free ( & reply ) ;
goto done ;
}
* auth_msg = reply ;
status = NT_STATUS_OK ;
done :
ntlmssp_end ( & ntlmssp_state ) ;
return status ;
}
static BOOL check_client_uid ( struct winbindd_cli_state * state , uid_t uid )
{
int ret ;
uid_t ret_uid ;
ret_uid = ( uid_t ) - 1 ;
2006-08-20 20:05:49 +00:00
ret = sys_getpeereid ( state - > sock , & ret_uid ) ;
2006-08-19 01:04:54 +00:00
if ( ret ! = 0 ) {
DEBUG ( 1 , ( " check_client_uid: Could not get socket peer uid: %s; "
" denying access \n " , strerror ( errno ) ) ) ;
return False ;
}
if ( uid ! = ret_uid ) {
DEBUG ( 1 , ( " check_client_uid: Client lied about its uid: said %d, "
" actually was %d; denying access \n " ,
uid , ret_uid ) ) ;
return False ;
}
return True ;
}
void winbindd_ccache_ntlm_auth ( struct winbindd_cli_state * state )
{
struct winbindd_domain * domain ;
fstring name_domain , name_user ;
/* Ensure null termination */
state - > request . data . ccache_ntlm_auth . user [
sizeof ( state - > request . data . ccache_ntlm_auth . user ) - 1 ] = ' \0 ' ;
DEBUG ( 3 , ( " [%5lu]: perform NTLM auth on behalf of user %s \n " , ( unsigned long ) state - > pid ,
state - > request . data . ccache_ntlm_auth . user ) ) ;
/* Parse domain and username */
if ( ! parse_domain_user ( state - > request . data . ccache_ntlm_auth . user ,
name_domain , name_user ) ) {
DEBUG ( 5 , ( " winbindd_ccache_ntlm_auth: cannot parse domain and user from name [%s] \n " ,
state - > request . data . ccache_ntlm_auth . user ) ) ;
request_error ( state ) ;
return ;
}
domain = find_auth_domain ( state , name_domain ) ;
if ( domain = = NULL ) {
DEBUG ( 5 , ( " winbindd_ccache_ntlm_auth: can't get domain [%s] \n " ,
name_domain ) ) ;
request_error ( state ) ;
return ;
}
if ( ! check_client_uid ( state , state - > request . data . ccache_ntlm_auth . uid ) ) {
request_error ( state ) ;
return ;
}
sendto_domain ( state , domain ) ;
}
enum winbindd_result winbindd_dual_ccache_ntlm_auth ( struct winbindd_domain * domain ,
struct winbindd_cli_state * state )
{
NTSTATUS result = NT_STATUS_NOT_SUPPORTED ;
2006-08-28 23:01:30 +00:00
struct WINBINDD_MEMORY_CREDS * entry ;
2006-08-19 01:04:54 +00:00
DATA_BLOB initial , challenge , auth ;
fstring name_domain , name_user ;
uint32 initial_blob_len , challenge_blob_len , extra_len ;
/* Ensure null termination */
state - > request . data . ccache_ntlm_auth . user [
sizeof ( state - > request . data . ccache_ntlm_auth . user ) - 1 ] = ' \0 ' ;
DEBUG ( 3 , ( " winbindd_dual_ccache_ntlm_auth: [%5lu]: perform NTLM auth on "
" behalf of user %s (dual) \n " , ( unsigned long ) state - > pid ,
state - > request . data . ccache_ntlm_auth . user ) ) ;
/* validate blob lengths */
initial_blob_len = state - > request . data . ccache_ntlm_auth . initial_blob_len ;
challenge_blob_len = state - > request . data . ccache_ntlm_auth . challenge_blob_len ;
extra_len = state - > request . extra_len ;
if ( initial_blob_len > extra_len | | challenge_blob_len > extra_len | |
initial_blob_len + challenge_blob_len > extra_len | |
initial_blob_len + challenge_blob_len < initial_blob_len | |
initial_blob_len + challenge_blob_len < challenge_blob_len ) {
DEBUG ( 10 , ( " winbindd_dual_ccache_ntlm_auth: blob lengths overrun "
" or wrap. Buffer [%d+%d > %d] \n " ,
initial_blob_len ,
challenge_blob_len ,
extra_len ) ) ;
goto process_result ;
}
/* Parse domain and username */
if ( ! parse_domain_user ( state - > request . data . ccache_ntlm_auth . user , name_domain , name_user ) ) {
DEBUG ( 10 , ( " winbindd_dual_ccache_ntlm_auth: cannot parse "
" domain and user from name [%s] \n " ,
state - > request . data . ccache_ntlm_auth . user ) ) ;
goto process_result ;
}
2006-08-28 23:01:30 +00:00
entry = find_memory_creds_by_name ( state - > request . data . ccache_ntlm_auth . user ) ;
if ( entry = = NULL | | entry - > nt_hash = = NULL | | entry - > lm_hash = = NULL ) {
2006-08-19 01:04:54 +00:00
DEBUG ( 10 , ( " winbindd_dual_ccache_ntlm_auth: could not find "
" credentials for user %s \n " ,
state - > request . data . ccache_ntlm_auth . user ) ) ;
goto process_result ;
}
2006-08-28 23:01:30 +00:00
DEBUG ( 10 , ( " winbindd_dual_ccache_ntlm_auth: found ccache [%s] \n " , entry - > username ) ) ;
2006-08-19 01:04:54 +00:00
if ( ! client_can_access_ccache_entry ( state - > request . data . ccache_ntlm_auth . uid , entry ) ) {
goto process_result ;
}
if ( initial_blob_len = = 0 & & challenge_blob_len = = 0 ) {
/* this is just a probe to see if credentials are available. */
result = NT_STATUS_OK ;
state - > response . data . ccache_ntlm_auth . auth_blob_len = 0 ;
goto process_result ;
}
initial = data_blob ( state - > request . extra_data . data , initial_blob_len ) ;
challenge = data_blob ( state - > request . extra_data . data + initial_blob_len ,
state - > request . data . ccache_ntlm_auth . challenge_blob_len ) ;
if ( ! initial . data | | ! challenge . data ) {
result = NT_STATUS_NO_MEMORY ;
} else {
2006-08-20 01:25:26 +00:00
result = do_ntlm_auth_with_hashes ( name_user , name_domain ,
2006-08-28 23:01:30 +00:00
entry - > lm_hash , entry - > nt_hash ,
2006-08-19 01:04:54 +00:00
initial , challenge , & auth ) ;
}
data_blob_free ( & initial ) ;
data_blob_free ( & challenge ) ;
if ( ! NT_STATUS_IS_OK ( result ) ) {
goto process_result ;
}
state - > response . extra_data . data = smb_xmemdup ( auth . data , auth . length ) ;
if ( ! state - > response . extra_data . data ) {
result = NT_STATUS_NO_MEMORY ;
goto process_result ;
}
state - > response . length + = auth . length ;
state - > response . data . ccache_ntlm_auth . auth_blob_len = auth . length ;
data_blob_free ( & auth ) ;
process_result :
return NT_STATUS_IS_OK ( result ) ? WINBINDD_OK : WINBINDD_ERROR ;
}