2010-04-12 00:37:08 +04:00
/*
Unix SMB / CIFS implementation .
Authentication utility functions
Copyright ( C ) Volker Lendecke 2010
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 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 , see < http : //www.gnu.org/licenses/>.
*/
# include "includes.h"
2011-03-25 04:28:05 +03:00
# include "auth.h"
2010-05-18 02:16:40 +04:00
# include "../lib/crypto/arcfour.h"
2010-08-05 17:14:04 +04:00
# include "../librpc/gen_ndr/netlogon.h"
2010-10-12 08:27:50 +04:00
# include "../libcli/security/security.h"
2011-01-11 17:08:41 +03:00
# include "rpc_client/util_netlogon.h"
2011-02-25 00:30:16 +03:00
# include "nsswitch/libwbclient/wbclient.h"
2013-12-13 22:08:34 +04:00
# include "lib/winbind_util.h"
2011-03-18 20:58:37 +03:00
# include "passdb.h"
2010-04-12 00:37:08 +04:00
# undef DBGC_CLASS
# define DBGC_CLASS DBGC_AUTH
/***************************************************************************
Make a server_info struct . Free with TALLOC_FREE ( ) .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
struct auth_serversupplied_info * make_server_info ( TALLOC_CTX * mem_ctx )
{
struct auth_serversupplied_info * result ;
2011-06-07 05:44:43 +04:00
result = talloc_zero ( mem_ctx , struct auth_serversupplied_info ) ;
2010-04-12 00:37:08 +04:00
if ( result = = NULL ) {
DEBUG ( 0 , ( " talloc failed \n " ) ) ;
return NULL ;
}
/* Initialise the uid and gid values to something non-zero
which may save us from giving away root access if there
is a bug in allocating these fields . */
result - > utok . uid = - 1 ;
result - > utok . gid = - 1 ;
2010-05-28 21:18:13 +04:00
2010-04-12 00:37:08 +04:00
return result ;
}
2010-04-11 17:25:55 +04:00
/****************************************************************************
2010-05-27 11:41:56 +04:00
inits a netr_SamInfo2 structure from an auth_serversupplied_info . sam2 must
already be initialized and is used as the talloc parent for its members .
2010-04-11 17:25:55 +04:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2010-05-27 11:41:56 +04:00
NTSTATUS serverinfo_to_SamInfo2 ( struct auth_serversupplied_info * server_info ,
struct netr_SamInfo2 * sam2 )
2010-04-11 17:25:55 +04:00
{
2010-05-27 11:41:56 +04:00
struct netr_SamInfo3 * info3 ;
2010-04-11 17:25:55 +04:00
2010-05-27 11:41:56 +04:00
info3 = copy_netr_SamInfo3 ( sam2 , server_info - > info3 ) ;
if ( ! info3 ) {
2010-04-11 17:25:55 +04:00
return NT_STATUS_NO_MEMORY ;
}
2011-02-14 03:35:21 +03:00
if ( server_info - > session_key . length ) {
2010-05-27 11:41:56 +04:00
memcpy ( info3 - > base . key . key ,
2011-02-14 03:35:21 +03:00
server_info - > session_key . data ,
2010-05-27 11:41:56 +04:00
MIN ( sizeof ( info3 - > base . key . key ) ,
2011-02-14 03:35:21 +03:00
server_info - > session_key . length ) ) ;
2010-04-11 17:25:55 +04:00
}
if ( server_info - > lm_session_key . length ) {
2010-05-27 11:41:56 +04:00
memcpy ( info3 - > base . LMSessKey . key ,
2010-04-11 17:25:55 +04:00
server_info - > lm_session_key . data ,
2010-05-27 11:41:56 +04:00
MIN ( sizeof ( info3 - > base . LMSessKey . key ) ,
2010-04-11 17:25:55 +04:00
server_info - > lm_session_key . length ) ) ;
}
2010-05-27 11:41:56 +04:00
sam2 - > base = info3 - > base ;
2010-04-11 17:25:55 +04:00
return NT_STATUS_OK ;
}
/****************************************************************************
inits a netr_SamInfo3 structure from an auth_serversupplied_info . sam3 must
already be initialized and is used as the talloc parent for its members .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2010-09-15 19:24:08 +04:00
NTSTATUS serverinfo_to_SamInfo3 ( const struct auth_serversupplied_info * server_info ,
2010-04-11 17:25:55 +04:00
struct netr_SamInfo3 * sam3 )
{
2010-05-27 11:41:56 +04:00
struct netr_SamInfo3 * info3 ;
info3 = copy_netr_SamInfo3 ( sam3 , server_info - > info3 ) ;
if ( ! info3 ) {
return NT_STATUS_NO_MEMORY ;
2010-04-11 17:25:55 +04:00
}
2011-02-14 03:35:21 +03:00
if ( server_info - > session_key . length ) {
2010-05-27 11:41:56 +04:00
memcpy ( info3 - > base . key . key ,
2011-02-14 03:35:21 +03:00
server_info - > session_key . data ,
2010-05-27 11:41:56 +04:00
MIN ( sizeof ( info3 - > base . key . key ) ,
2011-02-14 03:35:21 +03:00
server_info - > session_key . length ) ) ;
2010-05-27 11:41:56 +04:00
}
if ( server_info - > lm_session_key . length ) {
memcpy ( info3 - > base . LMSessKey . key ,
server_info - > lm_session_key . data ,
MIN ( sizeof ( info3 - > base . LMSessKey . key ) ,
server_info - > lm_session_key . length ) ) ;
}
sam3 - > base = info3 - > base ;
2010-04-11 17:25:55 +04:00
sam3 - > sidcount = 0 ;
sam3 - > sids = NULL ;
return NT_STATUS_OK ;
}
/****************************************************************************
inits a netr_SamInfo6 structure from an auth_serversupplied_info . sam6 must
already be initialized and is used as the talloc parent for its members .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
NTSTATUS serverinfo_to_SamInfo6 ( struct auth_serversupplied_info * server_info ,
struct netr_SamInfo6 * sam6 )
{
struct pdb_domain_info * dominfo ;
2010-05-27 11:41:56 +04:00
struct netr_SamInfo3 * info3 ;
2010-04-11 17:25:55 +04:00
if ( ( pdb_capabilities ( ) & PDB_CAP_ADS ) = = 0 ) {
DEBUG ( 10 , ( " Not adding validation info level 6 "
" without ADS passdb backend \n " ) ) ;
return NT_STATUS_INVALID_INFO_CLASS ;
}
dominfo = pdb_get_domain_info ( sam6 ) ;
if ( dominfo = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
2010-05-27 11:41:56 +04:00
info3 = copy_netr_SamInfo3 ( sam6 , server_info - > info3 ) ;
if ( ! info3 ) {
return NT_STATUS_NO_MEMORY ;
2010-04-11 17:25:55 +04:00
}
2011-02-14 03:35:21 +03:00
if ( server_info - > session_key . length ) {
2010-05-27 11:41:56 +04:00
memcpy ( info3 - > base . key . key ,
2011-02-14 03:35:21 +03:00
server_info - > session_key . data ,
2010-05-27 11:41:56 +04:00
MIN ( sizeof ( info3 - > base . key . key ) ,
2011-02-14 03:35:21 +03:00
server_info - > session_key . length ) ) ;
2010-05-27 11:41:56 +04:00
}
if ( server_info - > lm_session_key . length ) {
memcpy ( info3 - > base . LMSessKey . key ,
server_info - > lm_session_key . data ,
MIN ( sizeof ( info3 - > base . LMSessKey . key ) ,
server_info - > lm_session_key . length ) ) ;
}
sam6 - > base = info3 - > base ;
2010-04-11 17:25:55 +04:00
sam6 - > sidcount = 0 ;
sam6 - > sids = NULL ;
2010-04-12 16:49:14 +04:00
sam6 - > dns_domainname . string = talloc_strdup ( sam6 , dominfo - > dns_domain ) ;
if ( sam6 - > dns_domainname . string = = NULL ) {
2010-04-11 17:25:55 +04:00
return NT_STATUS_NO_MEMORY ;
}
sam6 - > principle . string = talloc_asprintf ( sam6 , " %s@%s " ,
2010-05-27 11:41:56 +04:00
sam6 - > base . account_name . string ,
2010-04-12 16:49:14 +04:00
sam6 - > dns_domainname . string ) ;
2010-04-11 17:25:55 +04:00
if ( sam6 - > principle . string = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
return NT_STATUS_OK ;
}
2010-05-27 10:07:33 +04:00
2010-05-28 21:18:13 +04:00
static NTSTATUS append_netr_SidAttr ( TALLOC_CTX * mem_ctx ,
struct netr_SidAttr * * sids ,
uint32_t * count ,
const struct dom_sid2 * asid ,
uint32_t attributes )
{
uint32_t t = * count ;
* sids = talloc_realloc ( mem_ctx , * sids , struct netr_SidAttr , t + 1 ) ;
if ( * sids = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
2010-08-26 19:21:39 +04:00
( * sids ) [ t ] . sid = dom_sid_dup ( * sids , asid ) ;
2010-05-28 21:18:13 +04:00
if ( ( * sids ) [ t ] . sid = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
( * sids ) [ t ] . attributes = attributes ;
* count = t + 1 ;
return NT_STATUS_OK ;
}
2010-11-05 17:44:55 +03:00
/* Fills the samr_RidWithAttributeArray with the provided sids.
2010-05-28 21:18:13 +04:00
* If it happens that we have additional groups that do not belong
* to the domain , add their sids as extra sids */
static NTSTATUS group_sids_to_info3 ( struct netr_SamInfo3 * info3 ,
const struct dom_sid * sids ,
size_t num_sids )
2010-05-27 10:07:33 +04:00
{
2010-05-28 21:18:13 +04:00
uint32_t attributes = SE_GROUP_MANDATORY |
SE_GROUP_ENABLED_BY_DEFAULT |
SE_GROUP_ENABLED ;
struct samr_RidWithAttributeArray * groups ;
struct dom_sid * domain_sid ;
2010-05-27 10:07:33 +04:00
unsigned int i ;
2010-05-28 21:18:13 +04:00
NTSTATUS status ;
uint32_t rid ;
2010-05-27 10:07:33 +04:00
bool ok ;
2010-05-28 21:18:13 +04:00
domain_sid = info3 - > base . domain_sid ;
groups = & info3 - > base . groups ;
groups - > rids = talloc_array ( info3 ,
2010-05-27 10:07:33 +04:00
struct samr_RidWithAttribute , num_sids ) ;
if ( ! groups - > rids ) {
return NT_STATUS_NO_MEMORY ;
}
for ( i = 0 ; i < num_sids ; i + + ) {
2010-05-28 21:18:13 +04:00
ok = sid_peek_check_rid ( domain_sid , & sids [ i ] , & rid ) ;
if ( ok ) {
/* store domain group rid */
2011-10-17 23:24:41 +04:00
groups - > rids [ groups - > count ] . rid = rid ;
groups - > rids [ groups - > count ] . attributes = attributes ;
2010-05-28 21:18:13 +04:00
groups - > count + + ;
continue ;
}
2010-05-27 10:07:33 +04:00
2010-05-28 21:18:13 +04:00
/* if this wasn't a domain sid, add it as extra sid */
status = append_netr_SidAttr ( info3 , & info3 - > sids ,
& info3 - > sidcount ,
& sids [ i ] , attributes ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
2010-05-27 10:07:33 +04:00
}
return NT_STATUS_OK ;
}
# define RET_NOMEM(ptr) do { \
if ( ! ptr ) { \
TALLOC_FREE ( info3 ) ; \
return NT_STATUS_NO_MEMORY ; \
} } while ( 0 )
NTSTATUS samu_to_SamInfo3 ( TALLOC_CTX * mem_ctx ,
struct samu * samu ,
const char * login_server ,
2010-05-28 21:18:13 +04:00
struct netr_SamInfo3 * * _info3 ,
struct extra_auth_info * extra )
2010-05-27 10:07:33 +04:00
{
struct netr_SamInfo3 * info3 ;
const struct dom_sid * user_sid ;
const struct dom_sid * group_sid ;
struct dom_sid domain_sid ;
struct dom_sid * group_sids ;
2011-02-21 12:30:28 +03:00
uint32_t num_group_sids = 0 ;
2010-05-27 10:07:33 +04:00
const char * tmp ;
gid_t * gids ;
NTSTATUS status ;
bool ok ;
user_sid = pdb_get_user_sid ( samu ) ;
group_sid = pdb_get_group_sid ( samu ) ;
if ( ! user_sid | | ! group_sid ) {
DEBUG ( 1 , ( " Sam account is missing sids! \n " ) ) ;
return NT_STATUS_UNSUCCESSFUL ;
}
info3 = talloc_zero ( mem_ctx , struct netr_SamInfo3 ) ;
if ( ! info3 ) {
return NT_STATUS_NO_MEMORY ;
}
2010-05-28 21:18:13 +04:00
ZERO_STRUCT ( domain_sid ) ;
/* check if this is a "Unix Users" domain user,
* we need to handle it in a special way if that ' s the case */
2010-11-05 17:48:44 +03:00
if ( sid_check_is_in_unix_users ( user_sid ) ) {
2010-05-28 21:18:13 +04:00
/* in info3 you can only set rids for the user and the
* primary group , and the domain sid must be that of
* the sam domain .
*
* Store a completely bogus value here .
* The real SID is stored in the extra sids .
* Other code will know to look there if ( - 1 ) is found
*/
info3 - > base . rid = ( uint32_t ) ( - 1 ) ;
sid_copy ( & extra - > user_sid , user_sid ) ;
DEBUG ( 10 , ( " Unix User found in struct samu. Rid marked as "
" special and sid (%s) saved as extra sid \n " ,
sid_string_dbg ( user_sid ) ) ) ;
} else {
sid_copy ( & domain_sid , user_sid ) ;
sid_split_rid ( & domain_sid , & info3 - > base . rid ) ;
}
if ( is_null_sid ( & domain_sid ) ) {
sid_copy ( & domain_sid , get_global_sam_sid ( ) ) ;
}
/* check if this is a "Unix Groups" domain group,
* if so we need special handling */
2010-11-05 17:48:44 +03:00
if ( sid_check_is_in_unix_groups ( group_sid ) ) {
2010-05-28 21:18:13 +04:00
/* in info3 you can only set rids for the user and the
* primary group , and the domain sid must be that of
* the sam domain .
*
* Store a completely bogus value here .
* The real SID is stored in the extra sids .
* Other code will know to look there if ( - 1 ) is found
*/
info3 - > base . primary_gid = ( uint32_t ) ( - 1 ) ;
sid_copy ( & extra - > pgid_sid , group_sid ) ;
DEBUG ( 10 , ( " Unix Group found in struct samu. Rid marked as "
" special and sid (%s) saved as extra sid \n " ,
sid_string_dbg ( group_sid ) ) ) ;
} else {
ok = sid_peek_check_rid ( & domain_sid , group_sid ,
& info3 - > base . primary_gid ) ;
if ( ! ok ) {
2010-07-24 15:17:15 +04:00
DEBUG ( 1 , ( " The primary group domain sid(%s) does not "
2010-05-28 21:18:13 +04:00
" match the domain sid(%s) for %s(%s) \n " ,
sid_string_dbg ( group_sid ) ,
sid_string_dbg ( & domain_sid ) ,
pdb_get_username ( samu ) ,
sid_string_dbg ( user_sid ) ) ) ;
TALLOC_FREE ( info3 ) ;
return NT_STATUS_UNSUCCESSFUL ;
}
}
2011-10-22 00:10:43 +04:00
unix_to_nt_time ( & info3 - > base . logon_time , pdb_get_logon_time ( samu ) ) ;
unix_to_nt_time ( & info3 - > base . logoff_time , get_time_t_max ( ) ) ;
unix_to_nt_time ( & info3 - > base . kickoff_time , get_time_t_max ( ) ) ;
2010-05-27 10:07:33 +04:00
unix_to_nt_time ( & info3 - > base . last_password_change ,
pdb_get_pass_last_set_time ( samu ) ) ;
unix_to_nt_time ( & info3 - > base . allow_password_change ,
pdb_get_pass_can_change_time ( samu ) ) ;
unix_to_nt_time ( & info3 - > base . force_password_change ,
pdb_get_pass_must_change_time ( samu ) ) ;
tmp = pdb_get_username ( samu ) ;
if ( tmp ) {
info3 - > base . account_name . string = talloc_strdup ( info3 , tmp ) ;
RET_NOMEM ( info3 - > base . account_name . string ) ;
}
tmp = pdb_get_fullname ( samu ) ;
if ( tmp ) {
info3 - > base . full_name . string = talloc_strdup ( info3 , tmp ) ;
RET_NOMEM ( info3 - > base . full_name . string ) ;
}
tmp = pdb_get_logon_script ( samu ) ;
if ( tmp ) {
info3 - > base . logon_script . string = talloc_strdup ( info3 , tmp ) ;
RET_NOMEM ( info3 - > base . logon_script . string ) ;
}
2010-06-11 16:25:27 +04:00
tmp = pdb_get_profile_path ( samu ) ;
2010-05-27 10:07:33 +04:00
if ( tmp ) {
info3 - > base . profile_path . string = talloc_strdup ( info3 , tmp ) ;
RET_NOMEM ( info3 - > base . profile_path . string ) ;
}
2010-06-11 16:25:27 +04:00
tmp = pdb_get_homedir ( samu ) ;
2010-05-27 10:07:33 +04:00
if ( tmp ) {
info3 - > base . home_directory . string = talloc_strdup ( info3 , tmp ) ;
RET_NOMEM ( info3 - > base . home_directory . string ) ;
}
2010-06-11 16:25:27 +04:00
tmp = pdb_get_dir_drive ( samu ) ;
2010-05-27 10:07:33 +04:00
if ( tmp ) {
info3 - > base . home_drive . string = talloc_strdup ( info3 , tmp ) ;
RET_NOMEM ( info3 - > base . home_drive . string ) ;
}
info3 - > base . logon_count = pdb_get_logon_count ( samu ) ;
info3 - > base . bad_password_count = pdb_get_bad_password_count ( samu ) ;
2011-10-22 00:10:43 +04:00
info3 - > base . logon_domain . string = talloc_strdup ( info3 ,
2011-09-16 17:45:47 +04:00
pdb_get_domain ( samu ) ) ;
2011-10-22 00:10:43 +04:00
RET_NOMEM ( info3 - > base . logon_domain . string ) ;
2011-09-16 17:45:47 +04:00
info3 - > base . domain_sid = dom_sid_dup ( info3 , & domain_sid ) ;
RET_NOMEM ( info3 - > base . domain_sid ) ;
2010-05-27 10:07:33 +04:00
status = pdb_enum_group_memberships ( mem_ctx , samu ,
& group_sids , & gids ,
& num_group_sids ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 1 , ( " Failed to get groups from sam account. \n " ) ) ;
TALLOC_FREE ( info3 ) ;
return status ;
}
2010-06-11 17:05:36 +04:00
if ( num_group_sids ) {
status = group_sids_to_info3 ( info3 , group_sids , num_group_sids ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
TALLOC_FREE ( info3 ) ;
return status ;
}
2010-05-27 10:07:33 +04:00
}
/* We don't need sids and gids after the conversion */
TALLOC_FREE ( group_sids ) ;
TALLOC_FREE ( gids ) ;
num_group_sids = 0 ;
/* FIXME: should we add other flags ? */
info3 - > base . user_flags = NETLOGON_EXTRA_SIDS ;
if ( login_server ) {
info3 - > base . logon_server . string = talloc_strdup ( info3 , login_server ) ;
RET_NOMEM ( info3 - > base . logon_server . string ) ;
}
info3 - > base . acct_flags = pdb_get_acct_ctrl ( samu ) ;
* _info3 = info3 ;
return NT_STATUS_OK ;
}
2013-12-13 22:08:34 +04:00
NTSTATUS passwd_to_SamInfo3 ( TALLOC_CTX * mem_ctx ,
const char * unix_username ,
const struct passwd * pwd ,
struct netr_SamInfo3 * * pinfo3 )
{
struct netr_SamInfo3 * info3 ;
NTSTATUS status ;
TALLOC_CTX * tmp_ctx ;
const char * domain_name = NULL ;
const char * user_name = NULL ;
struct dom_sid domain_sid ;
struct dom_sid user_sid ;
struct dom_sid group_sid ;
enum lsa_SidType type ;
uint32_t num_sids = 0 ;
struct dom_sid * user_sids = NULL ;
bool ok ;
tmp_ctx = talloc_stackframe ( ) ;
ok = lookup_name_smbconf ( tmp_ctx ,
unix_username ,
LOOKUP_NAME_ALL ,
& domain_name ,
& user_name ,
& user_sid ,
& type ) ;
if ( ! ok ) {
status = NT_STATUS_NO_SUCH_USER ;
goto done ;
}
if ( type ! = SID_NAME_USER ) {
status = NT_STATUS_NO_SUCH_USER ;
goto done ;
}
ok = winbind_lookup_usersids ( tmp_ctx ,
& user_sid ,
& num_sids ,
& user_sids ) ;
/* Check if winbind is running */
if ( ok ) {
/*
* Winbind is running and the first element of the user_sids
* is the primary group .
*/
if ( num_sids > 0 ) {
group_sid = user_sids [ 0 ] ;
}
} else {
/*
* Winbind is not running , create the group_sid from the
* group id .
*/
gid_to_sid ( & group_sid , pwd - > pw_gid ) ;
}
/* Make sure we have a valid group sid */
ok = ! is_null_sid ( & group_sid ) ;
if ( ! ok ) {
status = NT_STATUS_NO_SUCH_USER ;
goto done ;
}
/* Construct a netr_SamInfo3 from the information we have */
info3 = talloc_zero ( tmp_ctx , struct netr_SamInfo3 ) ;
if ( ! info3 ) {
status = NT_STATUS_NO_MEMORY ;
goto done ;
}
info3 - > base . account_name . string = talloc_strdup ( info3 , unix_username ) ;
if ( info3 - > base . account_name . string = = NULL ) {
status = NT_STATUS_NO_MEMORY ;
goto done ;
}
ZERO_STRUCT ( domain_sid ) ;
sid_copy ( & domain_sid , & user_sid ) ;
sid_split_rid ( & domain_sid , & info3 - > base . rid ) ;
info3 - > base . domain_sid = dom_sid_dup ( info3 , & domain_sid ) ;
ok = sid_peek_check_rid ( & domain_sid , & group_sid ,
& info3 - > base . primary_gid ) ;
if ( ! ok ) {
DEBUG ( 1 , ( " The primary group domain sid(%s) does not "
" match the domain sid(%s) for %s(%s) \n " ,
sid_string_dbg ( & group_sid ) ,
sid_string_dbg ( & domain_sid ) ,
unix_username ,
sid_string_dbg ( & user_sid ) ) ) ;
status = NT_STATUS_INVALID_SID ;
goto done ;
}
info3 - > base . acct_flags = ACB_NORMAL ;
if ( num_sids ) {
status = group_sids_to_info3 ( info3 , user_sids , num_sids ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
goto done ;
}
}
* pinfo3 = talloc_steal ( mem_ctx , info3 ) ;
status = NT_STATUS_OK ;
done :
talloc_free ( tmp_ctx ) ;
return status ;
}
2010-05-27 10:40:59 +04:00
# undef RET_NOMEM
# define RET_NOMEM(ptr) do { \
if ( ! ptr ) { \
TALLOC_FREE ( info3 ) ; \
return NULL ; \
} } while ( 0 )
struct netr_SamInfo3 * copy_netr_SamInfo3 ( TALLOC_CTX * mem_ctx ,
struct netr_SamInfo3 * orig )
{
struct netr_SamInfo3 * info3 ;
2010-05-28 21:35:56 +04:00
unsigned int i ;
2011-01-11 17:08:41 +03:00
NTSTATUS status ;
2010-05-27 10:40:59 +04:00
2010-05-28 03:22:02 +04:00
info3 = talloc_zero ( mem_ctx , struct netr_SamInfo3 ) ;
2010-05-27 10:40:59 +04:00
if ( ! info3 ) return NULL ;
2011-01-11 17:08:41 +03:00
status = copy_netr_SamBaseInfo ( info3 , & orig - > base , & info3 - > base ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
TALLOC_FREE ( info3 ) ;
return NULL ;
2010-05-28 03:22:02 +04:00
}
if ( orig - > sidcount ) {
2010-06-25 18:56:38 +04:00
info3 - > sidcount = orig - > sidcount ;
2010-05-28 21:35:56 +04:00
info3 - > sids = talloc_array ( info3 , struct netr_SidAttr ,
orig - > sidcount ) ;
2010-05-28 03:22:02 +04:00
RET_NOMEM ( info3 - > sids ) ;
2010-05-28 21:35:56 +04:00
for ( i = 0 ; i < orig - > sidcount ; i + + ) {
2010-08-26 19:21:39 +04:00
info3 - > sids [ i ] . sid = dom_sid_dup ( info3 - > sids ,
2010-05-28 21:35:56 +04:00
orig - > sids [ i ] . sid ) ;
RET_NOMEM ( info3 - > sids [ i ] . sid ) ;
2010-06-25 18:56:38 +04:00
info3 - > sids [ i ] . attributes =
orig - > sids [ i ] . attributes ;
2010-05-28 21:35:56 +04:00
}
2010-05-28 03:22:02 +04:00
}
2010-05-27 10:40:59 +04:00
return info3 ;
}
2010-05-27 11:21:35 +04:00
static NTSTATUS wbcsids_to_samr_RidWithAttributeArray (
TALLOC_CTX * mem_ctx ,
struct samr_RidWithAttributeArray * groups ,
const struct dom_sid * domain_sid ,
const struct wbcSidWithAttr * sids ,
size_t num_sids )
{
2012-02-01 20:02:52 +04:00
unsigned int i , j = 0 ;
2010-05-27 11:21:35 +04:00
bool ok ;
groups - > rids = talloc_array ( mem_ctx ,
struct samr_RidWithAttribute , num_sids ) ;
if ( ! groups - > rids ) {
return NT_STATUS_NO_MEMORY ;
}
/* a wbcDomainSid is the same as a dom_sid */
for ( i = 0 ; i < num_sids ; i + + ) {
ok = sid_peek_check_rid ( domain_sid ,
( const struct dom_sid * ) & sids [ i ] . sid ,
2012-02-01 20:02:52 +04:00
& groups - > rids [ j ] . rid ) ;
2010-05-27 11:21:35 +04:00
if ( ! ok ) continue ;
2012-02-01 20:02:52 +04:00
groups - > rids [ j ] . attributes = SE_GROUP_MANDATORY |
2010-05-27 11:21:35 +04:00
SE_GROUP_ENABLED_BY_DEFAULT |
SE_GROUP_ENABLED ;
2012-02-01 20:02:52 +04:00
j + + ;
2010-05-27 11:21:35 +04:00
}
2012-02-01 20:02:52 +04:00
groups - > count = j ;
2010-05-27 11:21:35 +04:00
return NT_STATUS_OK ;
}
2012-02-01 20:04:17 +04:00
static NTSTATUS wbcsids_to_netr_SidAttrArray (
const struct dom_sid * domain_sid ,
const struct wbcSidWithAttr * sids ,
size_t num_sids ,
TALLOC_CTX * mem_ctx ,
struct netr_SidAttr * * _info3_sids ,
uint32_t * info3_num_sids )
{
unsigned int i , j = 0 ;
struct netr_SidAttr * info3_sids ;
info3_sids = talloc_array ( mem_ctx , struct netr_SidAttr , num_sids ) ;
if ( info3_sids = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
/* a wbcDomainSid is the same as a dom_sid */
for ( i = 0 ; i < num_sids ; i + + ) {
const struct dom_sid * sid ;
sid = ( const struct dom_sid * ) & sids [ i ] . sid ;
if ( dom_sid_in_domain ( domain_sid , sid ) ) {
continue ;
}
info3_sids [ j ] . sid = dom_sid_dup ( info3_sids , sid ) ;
if ( info3_sids [ j ] . sid = = NULL ) {
talloc_free ( info3_sids ) ;
return NT_STATUS_NO_MEMORY ;
}
info3_sids [ j ] . attributes = SE_GROUP_MANDATORY |
SE_GROUP_ENABLED_BY_DEFAULT |
SE_GROUP_ENABLED ;
j + + ;
}
* info3_num_sids = j ;
* _info3_sids = info3_sids ;
return NT_STATUS_OK ;
}
2010-05-27 11:21:35 +04:00
struct netr_SamInfo3 * wbcAuthUserInfo_to_netr_SamInfo3 ( TALLOC_CTX * mem_ctx ,
const struct wbcAuthUserInfo * info )
{
struct netr_SamInfo3 * info3 ;
struct dom_sid user_sid ;
struct dom_sid group_sid ;
struct dom_sid domain_sid ;
NTSTATUS status ;
bool ok ;
memcpy ( & user_sid , & info - > sids [ 0 ] . sid , sizeof ( user_sid ) ) ;
memcpy ( & group_sid , & info - > sids [ 1 ] . sid , sizeof ( group_sid ) ) ;
info3 = talloc_zero ( mem_ctx , struct netr_SamInfo3 ) ;
if ( ! info3 ) return NULL ;
2013-01-31 16:39:42 +04:00
unix_to_nt_time ( & info3 - > base . logon_time , info - > logon_time ) ;
unix_to_nt_time ( & info3 - > base . logoff_time , info - > logoff_time ) ;
unix_to_nt_time ( & info3 - > base . kickoff_time , info - > kickoff_time ) ;
2010-12-18 18:02:09 +03:00
unix_to_nt_time ( & info3 - > base . last_password_change , info - > pass_last_set_time ) ;
unix_to_nt_time ( & info3 - > base . allow_password_change ,
info - > pass_can_change_time ) ;
unix_to_nt_time ( & info3 - > base . force_password_change ,
info - > pass_must_change_time ) ;
2010-05-27 11:21:35 +04:00
if ( info - > account_name ) {
info3 - > base . account_name . string =
talloc_strdup ( info3 , info - > account_name ) ;
RET_NOMEM ( info3 - > base . account_name . string ) ;
}
if ( info - > full_name ) {
info3 - > base . full_name . string =
talloc_strdup ( info3 , info - > full_name ) ;
RET_NOMEM ( info3 - > base . full_name . string ) ;
}
if ( info - > logon_script ) {
info3 - > base . logon_script . string =
talloc_strdup ( info3 , info - > logon_script ) ;
RET_NOMEM ( info3 - > base . logon_script . string ) ;
}
if ( info - > profile_path ) {
info3 - > base . profile_path . string =
talloc_strdup ( info3 , info - > profile_path ) ;
RET_NOMEM ( info3 - > base . profile_path . string ) ;
}
if ( info - > home_directory ) {
info3 - > base . home_directory . string =
talloc_strdup ( info3 , info - > home_directory ) ;
RET_NOMEM ( info3 - > base . home_directory . string ) ;
}
if ( info - > home_drive ) {
info3 - > base . home_drive . string =
talloc_strdup ( info3 , info - > home_drive ) ;
RET_NOMEM ( info3 - > base . home_drive . string ) ;
}
info3 - > base . logon_count = info - > logon_count ;
info3 - > base . bad_password_count = info - > bad_password_count ;
sid_copy ( & domain_sid , & user_sid ) ;
sid_split_rid ( & domain_sid , & info3 - > base . rid ) ;
ok = sid_peek_check_rid ( & domain_sid , & group_sid ,
& info3 - > base . primary_gid ) ;
if ( ! ok ) {
DEBUG ( 1 , ( " The primary group sid domain does not "
" match user sid domain for user: %s \n " ,
info - > account_name ) ) ;
TALLOC_FREE ( info3 ) ;
return NULL ;
}
status = wbcsids_to_samr_RidWithAttributeArray ( info3 ,
& info3 - > base . groups ,
& domain_sid ,
& info - > sids [ 1 ] ,
info - > num_sids - 1 ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
TALLOC_FREE ( info3 ) ;
return NULL ;
}
2012-02-01 20:04:17 +04:00
status = wbcsids_to_netr_SidAttrArray ( & domain_sid ,
& info - > sids [ 1 ] ,
info - > num_sids - 1 ,
info3 ,
& info3 - > sids ,
& info3 - > sidcount ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
TALLOC_FREE ( info3 ) ;
return NULL ;
}
2010-05-27 11:21:35 +04:00
info3 - > base . user_flags = info - > user_flags ;
memcpy ( info3 - > base . key . key , info - > user_session_key , 16 ) ;
if ( info - > logon_server ) {
info3 - > base . logon_server . string =
talloc_strdup ( info3 , info - > logon_server ) ;
RET_NOMEM ( info3 - > base . logon_server . string ) ;
}
if ( info - > domain_name ) {
2011-10-22 00:10:43 +04:00
info3 - > base . logon_domain . string =
2010-05-27 11:21:35 +04:00
talloc_strdup ( info3 , info - > domain_name ) ;
2011-10-22 00:10:43 +04:00
RET_NOMEM ( info3 - > base . logon_domain . string ) ;
2010-05-27 11:21:35 +04:00
}
2010-08-26 19:21:39 +04:00
info3 - > base . domain_sid = dom_sid_dup ( info3 , & domain_sid ) ;
2010-05-27 11:21:35 +04:00
RET_NOMEM ( info3 - > base . domain_sid ) ;
memcpy ( info3 - > base . LMSessKey . key , info - > lm_session_key , 8 ) ;
info3 - > base . acct_flags = info - > acct_flags ;
return info3 ;
}