1998-03-11 21:11:04 +00:00
/*
* Unix SMB / Netbios implementation .
* Version 1.9 .
* RPC Pipe client / server routines
* Copyright ( C ) Andrew Tridgell 1992 - 1997 ,
* Copyright ( C ) Luke Kenneth Casson Leighton 1996 - 1997 ,
* Copyright ( C ) Paul Ashton 1997.
1998-05-14 01:30:40 +00:00
* Copyright ( C ) Jeremy Allison 1998.
*
1998-03-11 21:11:04 +00:00
* 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 "nterr.h"
extern int DEBUGLEVEL ;
extern BOOL sam_logon_in_ssb ;
extern pstring samlogon_user ;
1998-04-25 01:12:08 +00:00
extern pstring global_myname ;
1998-03-11 21:11:04 +00:00
/*************************************************************************
make_net_r_req_chal :
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static void make_net_r_req_chal ( NET_R_REQ_CHAL * r_c ,
DOM_CHAL * srv_chal , int status )
{
DEBUG ( 6 , ( " make_net_r_req_chal: %d \n " , __LINE__ ) ) ;
memcpy ( r_c - > srv_chal . data , srv_chal - > data , sizeof ( srv_chal - > data ) ) ;
r_c - > status = status ;
}
/*************************************************************************
net_reply_req_chal :
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static void net_reply_req_chal ( NET_Q_REQ_CHAL * q_c , prs_struct * rdata ,
DOM_CHAL * srv_chal , uint32 srv_time )
{
NET_R_REQ_CHAL r_c ;
DEBUG ( 6 , ( " net_reply_req_chal: %d \n " , __LINE__ ) ) ;
/* set up the LSA REQUEST CHALLENGE response */
make_net_r_req_chal ( & r_c , srv_chal , srv_time ) ;
/* store the response in the SMB stream */
net_io_r_req_chal ( " " , & r_c , rdata , 0 ) ;
DEBUG ( 6 , ( " net_reply_req_chal: %d \n " , __LINE__ ) ) ;
}
/*************************************************************************
net_reply_logon_ctrl2 :
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static void net_reply_logon_ctrl2 ( NET_Q_LOGON_CTRL2 * q_l , prs_struct * rdata ,
uint32 flags , uint32 pdc_status , uint32 logon_attempts ,
uint32 tc_status , char * trust_domain_name )
{
NET_R_LOGON_CTRL2 r_l ;
DEBUG ( 6 , ( " net_reply_logon_ctrl2: %d \n " , __LINE__ ) ) ;
/* set up the Logon Control2 response */
make_r_logon_ctrl2 ( & r_l , q_l - > query_level ,
flags , pdc_status , logon_attempts ,
tc_status , trust_domain_name ) ;
/* store the response in the SMB stream */
net_io_r_logon_ctrl2 ( " " , & r_l , rdata , 0 ) ;
DEBUG ( 6 , ( " net_reply_logon_ctrl2: %d \n " , __LINE__ ) ) ;
}
/*************************************************************************
net_reply_trust_dom_list :
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static void net_reply_trust_dom_list ( NET_Q_TRUST_DOM_LIST * q_t , prs_struct * rdata ,
uint32 num_trust_domains , char * trust_domain_name )
{
NET_R_TRUST_DOM_LIST r_t ;
DEBUG ( 6 , ( " net_reply_trust_dom_list: %d \n " , __LINE__ ) ) ;
/* set up the Trusted Domain List response */
make_r_trust_dom ( & r_t , num_trust_domains , trust_domain_name ) ;
/* store the response in the SMB stream */
net_io_r_trust_dom ( " " , & r_t , rdata , 0 ) ;
DEBUG ( 6 , ( " net_reply_trust_dom_listlogon_ctrl2: %d \n " , __LINE__ ) ) ;
}
/*************************************************************************
make_net_r_auth_2 :
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static void make_net_r_auth_2 ( NET_R_AUTH_2 * r_a ,
DOM_CHAL * resp_cred , NEG_FLAGS * flgs , int status )
{
memcpy ( r_a - > srv_chal . data , resp_cred - > data , sizeof ( resp_cred - > data ) ) ;
memcpy ( & ( r_a - > srv_flgs ) , flgs , sizeof ( r_a - > srv_flgs ) ) ;
r_a - > status = status ;
}
/*************************************************************************
net_reply_auth_2 :
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static void net_reply_auth_2 ( NET_Q_AUTH_2 * q_a , prs_struct * rdata ,
DOM_CHAL * resp_cred , int status )
{
NET_R_AUTH_2 r_a ;
/* set up the LSA AUTH 2 response */
make_net_r_auth_2 ( & r_a , resp_cred , & ( q_a - > clnt_flgs ) , status ) ;
/* store the response in the SMB stream */
net_io_r_auth_2 ( " " , & r_a , rdata , 0 ) ;
}
/***********************************************************************************
make_net_r_srv_pwset :
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static void make_net_r_srv_pwset ( NET_R_SRV_PWSET * r_s ,
DOM_CRED * srv_cred , int status )
{
DEBUG ( 5 , ( " make_net_r_srv_pwset: %d \n " , __LINE__ ) ) ;
memcpy ( & ( r_s - > srv_cred ) , srv_cred , sizeof ( r_s - > srv_cred ) ) ;
r_s - > status = status ;
DEBUG ( 5 , ( " make_net_r_srv_pwset: %d \n " , __LINE__ ) ) ;
}
/*************************************************************************
net_reply_srv_pwset :
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static void net_reply_srv_pwset ( NET_Q_SRV_PWSET * q_s , prs_struct * rdata ,
DOM_CRED * srv_cred , int status )
{
NET_R_SRV_PWSET r_s ;
DEBUG ( 5 , ( " net_srv_pwset: %d \n " , __LINE__ ) ) ;
/* set up the LSA Server Password Set response */
make_net_r_srv_pwset ( & r_s , srv_cred , status ) ;
/* store the response in the SMB stream */
net_io_r_srv_pwset ( " " , & r_s , rdata , 0 ) ;
DEBUG ( 5 , ( " net_srv_pwset: %d \n " , __LINE__ ) ) ;
}
/*************************************************************************
net_reply_sam_logon :
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static void net_reply_sam_logon ( NET_Q_SAM_LOGON * q_s , prs_struct * rdata ,
DOM_CRED * srv_cred , NET_USER_INFO_3 * user_info ,
uint32 status )
{
NET_R_SAM_LOGON r_s ;
/* XXXX maybe we want to say 'no', reject the client's credentials */
r_s . buffer_creds = 1 ; /* yes, we have valid server credentials */
memcpy ( & ( r_s . srv_creds ) , srv_cred , sizeof ( r_s . srv_creds ) ) ;
/* store the user information, if there is any. */
r_s . user = user_info ;
if ( status = = 0x0 & & user_info ! = NULL & & user_info - > ptr_user_info ! = 0 )
{
r_s . switch_value = 3 ; /* indicates type of validation user info */
}
else
{
r_s . switch_value = 0 ; /* indicates no info */
}
r_s . status = status ;
r_s . auth_resp = 1 ; /* authoritative response */
/* store the response in the SMB stream */
net_io_r_sam_logon ( " " , & r_s , rdata , 0 ) ;
}
/*************************************************************************
net_reply_sam_logoff :
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static void net_reply_sam_logoff ( NET_Q_SAM_LOGOFF * q_s , prs_struct * rdata ,
DOM_CRED * srv_cred ,
uint32 status )
{
NET_R_SAM_LOGOFF r_s ;
/* XXXX maybe we want to say 'no', reject the client's credentials */
r_s . buffer_creds = 1 ; /* yes, we have valid server credentials */
memcpy ( & ( r_s . srv_creds ) , srv_cred , sizeof ( r_s . srv_creds ) ) ;
r_s . status = status ;
/* store the response in the SMB stream */
net_io_r_sam_logoff ( " " , & r_s , rdata , 0 ) ;
}
/******************************************************************
gets a machine password entry . checks access rights of the host .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static BOOL get_md4pw ( char * md4pw , char * mach_name , char * mach_acct )
{
1998-03-17 12:31:43 +00:00
struct smb_passwd * smb_pass ;
extern int Client ;
1998-03-11 21:11:04 +00:00
if ( ! allow_access ( lp_domain_hostsdeny ( ) , lp_domain_hostsallow ( ) ,
1998-03-17 12:31:43 +00:00
client_name ( Client ) , client_addr ( Client ) ) )
1998-03-11 21:11:04 +00:00
{
DEBUG ( 0 , ( " get_md4pw: Workstation %s denied access to domain \n " , mach_acct ) ) ;
return False ;
}
become_root ( True ) ;
1998-05-07 18:19:05 +00:00
smb_pass = getsampwnam ( mach_acct ) ;
1998-03-11 21:11:04 +00:00
unbecome_root ( True ) ;
if ( smb_pass ! = NULL )
{
memcpy ( md4pw , smb_pass - > smb_nt_passwd , 16 ) ;
dump_data ( 5 , md4pw , 16 ) ;
return True ;
}
DEBUG ( 0 , ( " get_md4pw: Workstation %s: no account in domain \n " , mach_acct ) ) ;
return False ;
}
/*************************************************************************
api_net_req_chal :
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static void api_net_req_chal ( int uid ,
prs_struct * data ,
prs_struct * rdata )
{
NET_Q_REQ_CHAL q_r ;
uint32 status = 0x0 ;
fstring mach_acct ;
fstring mach_name ;
user_struct * vuser ;
DEBUG ( 5 , ( " api_net_req_chal(%d): vuid %d \n " , __LINE__ , uid ) ) ;
if ( ( vuser = get_valid_user_struct ( uid ) ) = = NULL ) return ;
/* grab the challenge... */
net_io_q_req_chal ( " " , & q_r , data , 0 ) ;
fstrcpy ( mach_acct , unistrn2 ( q_r . uni_logon_clnt . buffer ,
q_r . uni_logon_clnt . uni_str_len ) ) ;
fstrcpy ( mach_name , mach_acct ) ;
strlower ( mach_name ) ;
1998-05-12 00:55:32 +00:00
fstrcat ( mach_acct , " $ " ) ;
1998-03-11 21:11:04 +00:00
1998-05-11 06:38:36 +00:00
if ( get_md4pw ( ( char * ) vuser - > dc . md4pw , mach_name , mach_acct ) )
1998-03-11 21:11:04 +00:00
{
/* copy the client credentials */
memcpy ( vuser - > dc . clnt_chal . data , q_r . clnt_chal . data , sizeof ( q_r . clnt_chal . data ) ) ;
memcpy ( vuser - > dc . clnt_cred . challenge . data , q_r . clnt_chal . data , sizeof ( q_r . clnt_chal . data ) ) ;
/* create a server challenge for the client */
1998-04-20 23:57:29 +00:00
/* Set these to random values. */
generate_random_buffer ( vuser - > dc . srv_chal . data , 8 , False ) ;
1998-03-11 21:11:04 +00:00
memcpy ( vuser - > dc . srv_cred . challenge . data , vuser - > dc . srv_chal . data , 8 ) ;
bzero ( vuser - > dc . sess_key , sizeof ( vuser - > dc . sess_key ) ) ;
/* from client / server challenges and md4 password, generate sess key */
cred_session_key ( & ( vuser - > dc . clnt_chal ) , & ( vuser - > dc . srv_chal ) ,
1998-05-11 06:38:36 +00:00
( char * ) vuser - > dc . md4pw , vuser - > dc . sess_key ) ;
1998-03-11 21:11:04 +00:00
}
else
{
/* lkclXXXX take a guess at a good error message to return :-) */
status = 0xC0000000 | NT_STATUS_NOLOGON_WORKSTATION_TRUST_ACCOUNT ;
}
/* construct reply. */
net_reply_req_chal ( & q_r , rdata ,
& ( vuser - > dc . srv_chal ) , status ) ;
}
/*************************************************************************
api_net_auth_2 :
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static void api_net_auth_2 ( int uid ,
prs_struct * data ,
prs_struct * rdata )
{
NET_Q_AUTH_2 q_a ;
uint32 status = 0x0 ;
DOM_CHAL srv_cred ;
UTIME srv_time ;
user_struct * vuser ;
if ( ( vuser = get_valid_user_struct ( uid ) ) = = NULL ) return ;
srv_time . time = 0 ;
/* grab the challenge... */
net_io_q_auth_2 ( " " , & q_a , data , 0 ) ;
/* check that the client credentials are valid */
if ( cred_assert ( & ( q_a . clnt_chal ) , vuser - > dc . sess_key ,
& ( vuser - > dc . clnt_cred . challenge ) , srv_time ) )
{
/* create server challenge for inclusion in the reply */
cred_create ( vuser - > dc . sess_key , & ( vuser - > dc . srv_cred . challenge ) , srv_time , & srv_cred ) ;
/* copy the received client credentials for use next time */
memcpy ( vuser - > dc . clnt_cred . challenge . data , & ( q_a . clnt_chal . data ) , sizeof ( q_a . clnt_chal . data ) ) ;
memcpy ( vuser - > dc . srv_cred . challenge . data , & ( q_a . clnt_chal . data ) , sizeof ( q_a . clnt_chal . data ) ) ;
}
else
{
status = NT_STATUS_ACCESS_DENIED | 0xC0000000 ;
}
/* construct reply. */
net_reply_auth_2 ( & q_a , rdata , & srv_cred , status ) ;
}
/*************************************************************************
api_net_srv_pwset :
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static void api_net_srv_pwset ( int uid ,
prs_struct * data ,
prs_struct * rdata )
{
NET_Q_SRV_PWSET q_a ;
uint32 status = NT_STATUS_WRONG_PASSWORD | 0xC0000000 ;
DOM_CRED srv_cred ;
1998-04-01 19:12:35 +00:00
pstring mach_acct ;
1998-03-11 21:11:04 +00:00
struct smb_passwd * smb_pass ;
BOOL ret ;
user_struct * vuser ;
if ( ( vuser = get_valid_user_struct ( uid ) ) = = NULL ) return ;
/* grab the challenge and encrypted password ... */
net_io_q_srv_pwset ( " " , & q_a , data , 0 ) ;
/* checks and updates credentials. creates reply credentials */
if ( deal_with_creds ( vuser - > dc . sess_key , & ( vuser - > dc . clnt_cred ) ,
1998-05-07 18:19:05 +00:00
& ( q_a . clnt_id . cred ) , & srv_cred ) )
1998-03-11 21:11:04 +00:00
{
memcpy ( & ( vuser - > dc . srv_cred ) , & ( vuser - > dc . clnt_cred ) , sizeof ( vuser - > dc . clnt_cred ) ) ;
DEBUG ( 5 , ( " api_net_srv_pwset: %d \n " , __LINE__ ) ) ;
1998-05-07 18:19:05 +00:00
pstrcpy ( mach_acct , unistrn2 ( q_a . clnt_id . login . uni_acct_name . buffer ,
q_a . clnt_id . login . uni_acct_name . uni_str_len ) ) ;
1998-03-11 21:11:04 +00:00
1998-05-07 18:19:05 +00:00
DEBUG ( 3 , ( " Server Password Set Wksta:[%s] \n " , mach_acct ) ) ;
1998-03-11 21:11:04 +00:00
1998-05-07 18:19:05 +00:00
become_root ( True ) ;
smb_pass = getsampwnam ( mach_acct ) ;
unbecome_root ( True ) ;
1998-03-11 21:11:04 +00:00
if ( smb_pass ! = NULL )
{
1998-05-07 18:19:05 +00:00
unsigned char pwd [ 16 ] ;
int i ;
1998-04-17 19:29:51 +00:00
1998-05-07 18:19:05 +00:00
DEBUG ( 100 , ( " Server password set : new given value was : \n " ) ) ;
for ( i = 0 ; i < 16 ; i + + )
{
DEBUG ( 100 , ( " %02X " , q_a . pwd [ i ] ) ) ;
}
DEBUG ( 100 , ( " \n " ) ) ;
1998-04-17 19:29:51 +00:00
1998-05-07 18:19:05 +00:00
cred_hash3 ( pwd , q_a . pwd , vuser - > dc . sess_key , 0 ) ;
1998-04-17 19:29:51 +00:00
1998-05-07 18:19:05 +00:00
/* lies! nt and lm passwords are _not_ the same: don't care */
smb_pass - > smb_passwd = pwd ;
smb_pass - > smb_nt_passwd = pwd ;
smb_pass - > acct_ctrl = ACB_WSTRUST ;
1998-04-17 19:29:51 +00:00
1998-05-07 18:19:05 +00:00
become_root ( True ) ;
ret = mod_sampwd_entry ( smb_pass , False ) ;
unbecome_root ( True ) ;
1998-04-17 19:29:51 +00:00
1998-05-07 18:19:05 +00:00
if ( ret )
{
/* hooray! */
status = 0x0 ;
}
}
1998-03-11 21:11:04 +00:00
DEBUG ( 5 , ( " api_net_srv_pwset: %d \n " , __LINE__ ) ) ;
}
else
{
/* lkclXXXX take a guess at a sensible error code to return... */
status = 0xC0000000 | NT_STATUS_NETWORK_CREDENTIAL_CONFLICT ;
}
1998-04-17 19:29:51 +00:00
/* Construct reply. */
net_reply_srv_pwset ( & q_a , rdata , & srv_cred , status ) ;
1998-03-11 21:11:04 +00:00
}
/*************************************************************************
api_net_sam_logoff :
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static void api_net_sam_logoff ( int uid ,
prs_struct * data ,
prs_struct * rdata )
{
NET_Q_SAM_LOGOFF q_l ;
NET_ID_INFO_CTR ctr ;
DOM_CRED srv_cred ;
user_struct * vuser ;
if ( ( vuser = get_valid_user_struct ( uid ) ) = = NULL ) return ;
/* the DOM_ID_INFO_1 structure is a bit big. plus we might want to
dynamically allocate it inside net_io_q_sam_logon , at some point */
q_l . sam_id . ctr = & ctr ;
/* grab the challenge... */
net_io_q_sam_logoff ( " " , & q_l , data , 0 ) ;
/* checks and updates credentials. creates reply credentials */
deal_with_creds ( vuser - > dc . sess_key , & ( vuser - > dc . clnt_cred ) ,
& ( q_l . sam_id . client . cred ) , & srv_cred ) ;
memcpy ( & ( vuser - > dc . srv_cred ) , & ( vuser - > dc . clnt_cred ) , sizeof ( vuser - > dc . clnt_cred ) ) ;
/* construct reply. always indicate success */
net_reply_sam_logoff ( & q_l , rdata ,
& srv_cred ,
0x0 ) ;
}
/*************************************************************************
net_login_interactive :
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static uint32 net_login_interactive ( NET_ID_INFO_1 * id1 ,
struct smb_passwd * smb_pass ,
user_struct * vuser )
{
uint32 status = 0x0 ;
char nt_pwd [ 16 ] ;
char lm_pwd [ 16 ] ;
1998-03-24 00:37:53 +00:00
unsigned char key [ 16 ] ;
cli_login.c :
start to create the calls needed for client-side of "network" logins,
which will be used for domain version of pass-through authentication.
unfortunately, none of this code is called in the main branch, because
smbclient BRANCH_NTDOM code isn't in use, yet!
srv_netlog.c :
fixed a problem with static net_login_network() which was being
stubborn: the if() statement looked horrendous, so i split it into
two if() statements, one to deal with the lm password; the other to
deal with the nt password.
the smb_password_check() functions should _not_ be called here, in
case we decide to disable lm hashes for security reasons, just like
nt does.
so, we now get a response to the SAM_LOGON "network" call, but the
connection is still dropped. a trace is needed from an nt server,
which is why i started on the client-side code. see above, which
is why i'm calling it a day :-)
(This used to be commit 2c29a7d9cf2d8b35e4b6e37e5d24caa91af3a9be)
1998-04-01 21:31:06 +00:00
1998-03-24 00:37:53 +00:00
memset ( key , 0 , 16 ) ;
memcpy ( key , vuser - > dc . sess_key , 8 ) ;
1998-03-11 21:11:04 +00:00
cli_login.c :
start to create the calls needed for client-side of "network" logins,
which will be used for domain version of pass-through authentication.
unfortunately, none of this code is called in the main branch, because
smbclient BRANCH_NTDOM code isn't in use, yet!
srv_netlog.c :
fixed a problem with static net_login_network() which was being
stubborn: the if() statement looked horrendous, so i split it into
two if() statements, one to deal with the lm password; the other to
deal with the nt password.
the smb_password_check() functions should _not_ be called here, in
case we decide to disable lm hashes for security reasons, just like
nt does.
so, we now get a response to the SAM_LOGON "network" call, but the
connection is still dropped. a trace is needed from an nt server,
which is why i started on the client-side code. see above, which
is why i'm calling it a day :-)
(This used to be commit 2c29a7d9cf2d8b35e4b6e37e5d24caa91af3a9be)
1998-04-01 21:31:06 +00:00
memcpy ( lm_pwd , id1 - > lm_owf . data , 16 ) ;
memcpy ( nt_pwd , id1 - > nt_owf . data , 16 ) ;
1998-05-11 06:38:36 +00:00
SamOEMhash ( ( uchar * ) lm_pwd , key , False ) ;
SamOEMhash ( ( uchar * ) nt_pwd , key , False ) ;
1998-03-11 21:11:04 +00:00
# ifdef DEBUG_PASSWORD
1998-03-24 00:37:53 +00:00
DEBUG ( 100 , ( " decrypt of lm owf password: " ) ) ;
1998-03-11 21:11:04 +00:00
dump_data ( 100 , lm_pwd , 16 ) ;
1998-03-24 00:37:53 +00:00
DEBUG ( 100 , ( " decrypt of nt owf password: " ) ) ;
1998-03-11 21:11:04 +00:00
dump_data ( 100 , nt_pwd , 16 ) ;
# endif
if ( memcmp ( smb_pass - > smb_passwd , lm_pwd , 16 ) ! = 0 & &
memcmp ( smb_pass - > smb_nt_passwd , nt_pwd , 16 ) ! = 0 )
{
status = 0xC0000000 | NT_STATUS_WRONG_PASSWORD ;
}
return status ;
}
/*************************************************************************
net_login_network :
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static uint32 net_login_network ( NET_ID_INFO_2 * id2 ,
struct smb_passwd * smb_pass ,
user_struct * vuser )
{
cli_login.c :
start to create the calls needed for client-side of "network" logins,
which will be used for domain version of pass-through authentication.
unfortunately, none of this code is called in the main branch, because
smbclient BRANCH_NTDOM code isn't in use, yet!
srv_netlog.c :
fixed a problem with static net_login_network() which was being
stubborn: the if() statement looked horrendous, so i split it into
two if() statements, one to deal with the lm password; the other to
deal with the nt password.
the smb_password_check() functions should _not_ be called here, in
case we decide to disable lm hashes for security reasons, just like
nt does.
so, we now get a response to the SAM_LOGON "network" call, but the
connection is still dropped. a trace is needed from an nt server,
which is why i started on the client-side code. see above, which
is why i'm calling it a day :-)
(This used to be commit 2c29a7d9cf2d8b35e4b6e37e5d24caa91af3a9be)
1998-04-01 21:31:06 +00:00
DEBUG ( 5 , ( " net_login_network: lm_len: %d nt_len: %d \n " ,
1998-04-29 00:02:57 +00:00
id2 - > hdr_lm_chal_resp . str_str_len ,
id2 - > hdr_nt_chal_resp . str_str_len ) ) ;
cli_login.c :
start to create the calls needed for client-side of "network" logins,
which will be used for domain version of pass-through authentication.
unfortunately, none of this code is called in the main branch, because
smbclient BRANCH_NTDOM code isn't in use, yet!
srv_netlog.c :
fixed a problem with static net_login_network() which was being
stubborn: the if() statement looked horrendous, so i split it into
two if() statements, one to deal with the lm password; the other to
deal with the nt password.
the smb_password_check() functions should _not_ be called here, in
case we decide to disable lm hashes for security reasons, just like
nt does.
so, we now get a response to the SAM_LOGON "network" call, but the
connection is still dropped. a trace is needed from an nt server,
which is why i started on the client-side code. see above, which
is why i'm calling it a day :-)
(This used to be commit 2c29a7d9cf2d8b35e4b6e37e5d24caa91af3a9be)
1998-04-01 21:31:06 +00:00
1998-04-02 01:07:53 +00:00
/* JRA. Check the NT password first if it exists - this is a higher quality
password , if it exists and it doesn ' t match - fail . */
1998-04-29 00:02:57 +00:00
if ( id2 - > hdr_nt_chal_resp . str_str_len = = 24 & &
1998-04-02 01:07:53 +00:00
smb_pass - > smb_nt_passwd ! = NULL )
{
1998-05-11 06:38:36 +00:00
if ( smb_password_check ( ( char * ) id2 - > nt_chal_resp . buffer ,
1998-04-02 01:07:53 +00:00
smb_pass - > smb_nt_passwd ,
id2 - > lm_chal ) )
return 0x0 ;
else
return 0xC0000000 | NT_STATUS_WRONG_PASSWORD ;
}
cli_login.c :
start to create the calls needed for client-side of "network" logins,
which will be used for domain version of pass-through authentication.
unfortunately, none of this code is called in the main branch, because
smbclient BRANCH_NTDOM code isn't in use, yet!
srv_netlog.c :
fixed a problem with static net_login_network() which was being
stubborn: the if() statement looked horrendous, so i split it into
two if() statements, one to deal with the lm password; the other to
deal with the nt password.
the smb_password_check() functions should _not_ be called here, in
case we decide to disable lm hashes for security reasons, just like
nt does.
so, we now get a response to the SAM_LOGON "network" call, but the
connection is still dropped. a trace is needed from an nt server,
which is why i started on the client-side code. see above, which
is why i'm calling it a day :-)
(This used to be commit 2c29a7d9cf2d8b35e4b6e37e5d24caa91af3a9be)
1998-04-01 21:31:06 +00:00
/* lkclXXXX this is not a good place to put disabling of LM hashes in.
if that is to be done , first move this entire function into a
library routine that calls the two smb_password_check ( ) functions .
if disabling LM hashes ( which nt can do for security reasons ) then
an attempt should be made to disable them everywhere ( which nt does
not do , for various security - hole reasons ) .
*/
1998-04-29 00:02:57 +00:00
if ( id2 - > hdr_lm_chal_resp . str_str_len = = 24 & &
1998-05-11 06:38:36 +00:00
smb_password_check ( ( char * ) id2 - > lm_chal_resp . buffer ,
cli_login.c :
start to create the calls needed for client-side of "network" logins,
which will be used for domain version of pass-through authentication.
unfortunately, none of this code is called in the main branch, because
smbclient BRANCH_NTDOM code isn't in use, yet!
srv_netlog.c :
fixed a problem with static net_login_network() which was being
stubborn: the if() statement looked horrendous, so i split it into
two if() statements, one to deal with the lm password; the other to
deal with the nt password.
the smb_password_check() functions should _not_ be called here, in
case we decide to disable lm hashes for security reasons, just like
nt does.
so, we now get a response to the SAM_LOGON "network" call, but the
connection is still dropped. a trace is needed from an nt server,
which is why i started on the client-side code. see above, which
is why i'm calling it a day :-)
(This used to be commit 2c29a7d9cf2d8b35e4b6e37e5d24caa91af3a9be)
1998-04-01 21:31:06 +00:00
smb_pass - > smb_passwd ,
id2 - > lm_chal ) )
{
return 0x0 ;
}
/* oops! neither password check succeeded */
1998-03-11 21:11:04 +00:00
return 0xC0000000 | NT_STATUS_WRONG_PASSWORD ;
}
/*************************************************************************
api_net_sam_logon :
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static void api_net_sam_logon ( int uid ,
prs_struct * data ,
prs_struct * rdata )
{
1998-05-14 01:30:40 +00:00
NET_Q_SAM_LOGON q_l ;
NET_ID_INFO_CTR ctr ;
NET_USER_INFO_3 usr_info ;
uint32 status = 0x0 ;
DOM_CRED srv_cred ;
struct smb_passwd * smb_pass = NULL ;
UNISTR2 * uni_samlogon_user = NULL ;
user_struct * vuser = NULL ;
if ( ( vuser = get_valid_user_struct ( uid ) ) = = NULL )
return ;
q_l . sam_id . ctr = & ctr ;
net_io_q_sam_logon ( " " , & q_l , data , 0 ) ;
/* checks and updates credentials. creates reply credentials */
if ( ! deal_with_creds ( vuser - > dc . sess_key , & ( vuser - > dc . clnt_cred ) ,
& ( q_l . sam_id . client . cred ) , & srv_cred ) )
{
status = 0xC0000000 | NT_STATUS_INVALID_HANDLE ;
}
else
{
memcpy ( & ( vuser - > dc . srv_cred ) , & ( vuser - > dc . clnt_cred ) , sizeof ( vuser - > dc . clnt_cred ) ) ;
}
/* find the username */
if ( status = = 0 )
{
switch ( q_l . sam_id . logon_level )
{
case INTERACTIVE_LOGON_TYPE :
{
uni_samlogon_user = & ( q_l . sam_id . ctr - > auth . id1 . uni_user_name ) ;
DEBUG ( 3 , ( " SAM Logon (Interactive). Domain:[%s]. " , lp_workgroup ( ) ) ) ;
break ;
}
case NET_LOGON_TYPE :
{
uni_samlogon_user = & ( q_l . sam_id . ctr - > auth . id2 . uni_user_name ) ;
DEBUG ( 3 , ( " SAM Logon (Network). Domain:[%s]. " , lp_workgroup ( ) ) ) ;
break ;
}
default :
{
DEBUG ( 2 , ( " SAM Logon: unsupported switch value \n " ) ) ;
status = 0xC0000000 | NT_STATUS_INVALID_INFO_CLASS ;
break ;
}
} /* end switch */
} /* end if status == 0 */
/* check username exists */
if ( status = = 0 )
{
pstrcpy ( samlogon_user , unistrn2 ( uni_samlogon_user - > buffer ,
uni_samlogon_user - > uni_str_len ) ) ;
DEBUG ( 3 , ( " User:[%s] \n " , samlogon_user ) ) ;
become_root ( True ) ;
smb_pass = getsampwnam ( samlogon_user ) ;
unbecome_root ( True ) ;
if ( smb_pass = = NULL )
{
status = 0xC0000000 | NT_STATUS_NO_SUCH_USER ;
}
}
/* validate password. */
if ( status = = 0 )
{
switch ( q_l . sam_id . logon_level )
{
case INTERACTIVE_LOGON_TYPE :
{
/* interactive login. */
status = net_login_interactive ( & q_l . sam_id . ctr - > auth . id1 , smb_pass , vuser ) ;
break ;
}
case NET_LOGON_TYPE :
{
/* network login. lm challenge and 24 byte responses */
status = net_login_network ( & q_l . sam_id . ctr - > auth . id2 , smb_pass , vuser ) ;
break ;
}
}
}
1998-03-11 21:11:04 +00:00
1998-05-14 01:30:40 +00:00
/* lkclXXXX this is the point at which, if the login was
successful , that the SAM Local Security Authority should
record that the user is logged in to the domain .
*/
/* return the profile plus other bits :-) */
if ( status = = 0 )
{
DOM_GID * gids = NULL ;
int num_gids = 0 ;
NTTIME dummy_time ;
pstring logon_script ;
pstring profile_path ;
pstring home_dir ;
pstring home_drive ;
pstring my_name ;
pstring my_workgroup ;
pstring domain_groups ;
DOM_SID dom_sid ;
char * other_sids ;
uint32 r_uid ;
uint32 r_gid ;
/* set up pointer indicating user/password failed to be found */
usr_info . ptr_user_info = 0 ;
dummy_time . low = 0xffffffff ;
dummy_time . high = 0x7fffffff ;
/* XXXX hack to get standard_sub_basic() to use sam logon username */
/* possibly a better way would be to do a become_user() call */
sam_logon_in_ssb = True ;
pstrcpy ( logon_script , lp_logon_script ( ) ) ;
pstrcpy ( profile_path , lp_logon_path ( ) ) ;
string_to_sid ( & dom_sid , lp_domain_sid ( ) ) ;
pstrcpy ( other_sids , lp_domain_other_sids ( ) ) ;
pstrcpy ( my_workgroup , lp_workgroup ( ) ) ;
pstrcpy ( home_drive , lp_logon_drive ( ) ) ;
pstrcpy ( home_dir , lp_logon_home ( ) ) ;
pstrcpy ( my_name , global_myname ) ;
strupper ( my_name ) ;
/*
* This is the point at which we get the group
* database - we should be getting the gid_t list
* from / etc / group and then turning the uids into
* rids and then into machine sids for this user .
* JRA .
*/
get_domain_user_groups ( domain_groups , samlogon_user ) ;
/*
* make_dom_gids allocates the gids array . JRA .
*/
gids = NULL ;
num_gids = make_dom_gids ( domain_groups , & gids ) ;
sam_logon_in_ssb = False ;
if ( name_to_rid ( samlogon_user , & r_uid , & r_gid ) )
{
make_net_user_info3 ( & usr_info ,
& dummy_time , /* logon_time */
& dummy_time , /* logoff_time */
& dummy_time , /* kickoff_time */
& dummy_time , /* pass_last_set_time */
& dummy_time , /* pass_can_change_time */
& dummy_time , /* pass_must_change_time */
samlogon_user , /* user_name */
vuser - > real_name , /* full_name */
logon_script , /* logon_script */
profile_path , /* profile_path */
home_dir , /* home_dir */
home_drive , /* dir_drive */
0 , /* logon_count */
0 , /* bad_pw_count */
r_uid , /* RID user_id */
r_gid , /* RID group_id */
num_gids , /* uint32 num_groups */
gids , /* DOM_GID *gids */
0x20 , /* uint32 user_flgs (?) */
NULL , /* char sess_key[16] */
my_name , /* char *logon_srv */
my_workgroup , /* char *logon_dom */
& dom_sid , /* DOM_SID *dom_sid */
other_sids ) ; /* char *other_sids */
}
else
{
status = 0xC0000000 | NT_STATUS_NO_SUCH_USER ;
}
/* Free any allocated groups array. */
if ( gids )
free ( ( char * ) gids ) ;
}
net_reply_sam_logon ( & q_l , rdata , & srv_cred , & usr_info , status ) ;
1998-03-11 21:11:04 +00:00
}
/*************************************************************************
api_net_trust_dom_list :
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static void api_net_trust_dom_list ( int uid ,
prs_struct * data ,
prs_struct * rdata )
{
NET_Q_TRUST_DOM_LIST q_t ;
char * trusted_domain = " test_domain " ;
DEBUG ( 6 , ( " api_net_trust_dom_list: %d \n " , __LINE__ ) ) ;
/* grab the lsa trusted domain list query... */
net_io_q_trust_dom ( " " , & q_t , data , 0 ) ;
/* construct reply. */
net_reply_trust_dom_list ( & q_t , rdata ,
1 , trusted_domain ) ;
DEBUG ( 6 , ( " api_net_trust_dom_list: %d \n " , __LINE__ ) ) ;
}
/*************************************************************************
error messages cropping up when using nltest . exe . . .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
# define ERROR_NO_SUCH_DOMAIN 0x54b
# define ERROR_NO_LOGON_SERVERS 0x51f
/*************************************************************************
api_net_logon_ctrl2 :
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static void api_net_logon_ctrl2 ( int uid ,
prs_struct * data ,
prs_struct * rdata )
{
NET_Q_LOGON_CTRL2 q_l ;
/* lkclXXXX - guess what - absolutely no idea what these are! */
uint32 flags = 0x0 ;
uint32 pdc_connection_status = 0x0 ;
uint32 logon_attempts = 0x0 ;
uint32 tc_status = ERROR_NO_LOGON_SERVERS ;
char * trusted_domain = " test_domain " ;
DEBUG ( 6 , ( " api_net_logon_ctrl2: %d \n " , __LINE__ ) ) ;
/* grab the lsa netlogon ctrl2 query... */
net_io_q_logon_ctrl2 ( " " , & q_l , data , 0 ) ;
/* construct reply. */
net_reply_logon_ctrl2 ( & q_l , rdata ,
flags , pdc_connection_status , logon_attempts ,
tc_status , trusted_domain ) ;
DEBUG ( 6 , ( " api_net_logon_ctrl2: %d \n " , __LINE__ ) ) ;
}
/*******************************************************************
array of \ PIPE \ NETLOGON operations
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static struct api_struct api_net_cmds [ ] =
{
{ " NET_REQCHAL " , NET_REQCHAL , api_net_req_chal } ,
{ " NET_AUTH2 " , NET_AUTH2 , api_net_auth_2 } ,
{ " NET_SRVPWSET " , NET_SRVPWSET , api_net_srv_pwset } ,
{ " NET_SAMLOGON " , NET_SAMLOGON , api_net_sam_logon } ,
{ " NET_SAMLOGOFF " , NET_SAMLOGOFF , api_net_sam_logoff } ,
{ " NET_LOGON_CTRL2 " , NET_LOGON_CTRL2 , api_net_logon_ctrl2 } ,
{ " NET_TRUST_DOM_LIST " , NET_TRUST_DOM_LIST , api_net_trust_dom_list } ,
{ NULL , 0 , NULL }
} ;
/*******************************************************************
receives a netlogon pipe and responds .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
BOOL api_netlog_rpc ( pipes_struct * p , prs_struct * data )
{
return api_rpcTNP ( p , " api_netlog_rpc " , api_net_cmds , data ) ;
}