2007-09-19 12:45:35 +00:00
/*
2002-01-30 06:08:46 +00:00
Unix SMB / CIFS implementation .
2000-12-13 12:52:21 +00:00
RPC pipe client
2001-05-07 01:55:08 +00:00
Copyright ( C ) Tim Potter 2000 - 2001 ,
2000-12-13 12:52:21 +00:00
Copyright ( C ) Andrew Tridgell 1992 - 1997 , 2000 ,
2002-05-17 13:49:01 +00:00
Copyright ( C ) Rafal Szczesniak 2002
2005-09-30 17:13:37 +00:00
Copyright ( C ) Jeremy Allison 2005.
2007-09-20 09:15:26 +00:00
Copyright ( C ) Michael Adam 2007.
2008-02-27 19:38:48 +01:00
Copyright ( C ) Guenther Deschner 2008.
2007-09-19 12:45:35 +00:00
2000-12-13 12:52:21 +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
2007-07-09 19:25:36 +00:00
the Free Software Foundation ; either version 3 of the License , or
2000-12-13 12:52:21 +00:00
( at your option ) any later version .
2007-09-19 12:45:35 +00:00
2000-12-13 12:52:21 +00:00
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 .
2007-09-19 12:45:35 +00:00
2000-12-13 12:52:21 +00:00
You should have received a copy of the GNU General Public License
2007-07-10 00:52:41 +00:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
2000-12-13 12:52:21 +00:00
*/
# include "includes.h"
2011-04-13 14:32:16 +02:00
# include "rpc_client/rpc_client.h"
2011-01-18 16:32:20 +01:00
# include "../librpc/gen_ndr/ndr_lsa_c.h"
2010-05-18 18:26:16 +02:00
# include "rpc_client/cli_lsarpc.h"
2010-08-19 23:15:22 +02:00
# include "rpc_client/init_lsa.h"
2010-10-12 15:27:50 +11:00
# include "../libcli/security/security.h"
2000-12-13 12:52:21 +00:00
2002-01-06 03:48:41 +00:00
/** @defgroup lsa LSA - Local Security Architecture
2001-11-23 05:50:05 +00:00
* @ ingroup rpc_client
2001-11-15 06:03:22 +00:00
*
* @ {
* */
/**
* @ file cli_lsarpc . c
*
* RPC client routines for the LSA RPC pipe . LSA means " local
* security authority " , which is half of a password database.
* */
2010-09-07 11:24:52 +02:00
NTSTATUS dcerpc_lsa_open_policy ( struct dcerpc_binding_handle * h ,
2005-06-08 22:10:34 +00:00
TALLOC_CTX * mem_ctx ,
2010-09-07 11:24:52 +02:00
bool sec_qos ,
uint32_t des_access ,
struct policy_handle * pol ,
NTSTATUS * result )
2000-12-13 12:52:21 +00:00
{
2008-02-11 17:50:18 +01:00
struct lsa_ObjectAttribute attr ;
struct lsa_QosInfo qos ;
uint16_t system_name = ' \\ ' ;
2000-12-13 12:52:21 +00:00
2008-12-08 18:03:01 +01:00
ZERO_STRUCT ( attr ) ;
attr . len = 0x18 ;
2000-12-13 12:52:21 +00:00
if ( sec_qos ) {
2008-12-08 18:03:01 +01:00
qos . len = 0xc ;
qos . impersonation_level = 2 ;
qos . context_mode = 1 ;
qos . effective_only = 0 ;
attr . sec_qos = & qos ;
2000-12-13 12:52:21 +00:00
}
2010-09-07 11:24:52 +02:00
return dcerpc_lsa_OpenPolicy ( h ,
mem_ctx ,
2008-02-11 17:50:18 +01:00
& system_name ,
& attr ,
des_access ,
2010-09-07 11:24:52 +02:00
pol ,
result ) ;
}
/** Open a LSA policy handle
*
* @ param cli Handle on an initialised SMB connection */
NTSTATUS rpccli_lsa_open_policy ( struct rpc_pipe_client * cli ,
TALLOC_CTX * mem_ctx ,
bool sec_qos , uint32 des_access ,
struct policy_handle * pol )
{
NTSTATUS status ;
NTSTATUS result = NT_STATUS_UNSUCCESSFUL ;
status = dcerpc_lsa_open_policy ( cli - > binding_handle ,
mem_ctx ,
sec_qos ,
des_access ,
pol ,
& result ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
return result ;
2000-12-13 12:52:21 +00:00
}
2010-09-07 12:19:25 +02:00
NTSTATUS dcerpc_lsa_open_policy2 ( struct dcerpc_binding_handle * h ,
TALLOC_CTX * mem_ctx ,
const char * srv_name_slash ,
bool sec_qos ,
uint32_t des_access ,
struct policy_handle * pol ,
NTSTATUS * result )
2001-08-21 03:05:27 +00:00
{
2008-02-11 17:50:18 +01:00
struct lsa_ObjectAttribute attr ;
struct lsa_QosInfo qos ;
2001-08-21 03:05:27 +00:00
2008-12-08 18:03:01 +01:00
ZERO_STRUCT ( attr ) ;
attr . len = 0x18 ;
2001-08-21 03:05:27 +00:00
if ( sec_qos ) {
2008-12-08 18:03:01 +01:00
qos . len = 0xc ;
qos . impersonation_level = 2 ;
qos . context_mode = 1 ;
qos . effective_only = 0 ;
attr . sec_qos = & qos ;
2001-08-21 03:05:27 +00:00
}
2010-09-07 12:19:25 +02:00
return dcerpc_lsa_OpenPolicy2 ( h ,
mem_ctx ,
srv_name_slash ,
2008-02-11 17:50:18 +01:00
& attr ,
des_access ,
2010-09-07 12:19:25 +02:00
pol ,
result ) ;
}
/** Open a LSA policy handle
*
* @ param cli Handle on an initialised SMB connection
*/
NTSTATUS rpccli_lsa_open_policy2 ( struct rpc_pipe_client * cli ,
TALLOC_CTX * mem_ctx , bool sec_qos ,
uint32 des_access , struct policy_handle * pol )
{
NTSTATUS status ;
NTSTATUS result = NT_STATUS_UNSUCCESSFUL ;
2011-03-23 16:47:26 +01:00
status = dcerpc_lsa_open_policy2 ( cli - > binding_handle ,
mem_ctx ,
2011-03-29 00:01:40 +02:00
cli - > srv_name_slash ,
2011-03-23 16:47:26 +01:00
sec_qos ,
des_access ,
pol ,
& result ) ;
2010-09-07 12:19:25 +02:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
return result ;
2001-08-21 03:05:27 +00:00
}
2007-05-22 11:30:35 +00:00
/* Lookup a list of sids
*
2007-05-22 13:40:01 +00:00
* internal version withOUT memory allocation of the target arrays .
2011-03-10 17:11:37 +01:00
* this assumes sufficiently sized arrays to store domains , names and types . */
2007-05-22 11:30:35 +00:00
2010-09-07 14:58:19 +02:00
static NTSTATUS dcerpc_lsa_lookup_sids_noalloc ( struct dcerpc_binding_handle * h ,
2007-05-22 11:30:35 +00:00
TALLOC_CTX * mem_ctx ,
2009-03-18 22:49:41 +01:00
struct policy_handle * pol ,
2007-05-22 11:30:35 +00:00
int num_sids ,
2010-05-21 11:25:01 +10:00
const struct dom_sid * sids ,
2007-05-22 11:30:35 +00:00
char * * domains ,
char * * names ,
2009-09-13 00:28:49 +02:00
enum lsa_SidType * types ,
2010-09-07 14:58:19 +02:00
bool use_lookupsids3 ,
NTSTATUS * presult )
2007-05-22 11:30:35 +00:00
{
2010-09-07 14:58:19 +02:00
NTSTATUS status = NT_STATUS_OK ;
NTSTATUS result = NT_STATUS_UNSUCCESSFUL ;
2008-02-18 10:03:19 +01:00
struct lsa_SidArray sid_array ;
struct lsa_RefDomainList * ref_domains = NULL ;
struct lsa_TransNameArray lsa_names ;
2010-09-07 14:58:19 +02:00
enum lsa_LookupNamesLevel level = LSA_LOOKUP_NAMES_ALL ;
2008-02-18 10:03:19 +01:00
uint32_t count = 0 ;
2010-09-07 14:58:19 +02:00
int i ;
2008-02-18 10:03:19 +01:00
ZERO_STRUCT ( lsa_names ) ;
2007-05-22 11:30:35 +00:00
2008-02-18 10:03:19 +01:00
sid_array . num_sids = num_sids ;
2011-06-07 11:30:12 +10:00
sid_array . sids = talloc_array ( mem_ctx , struct lsa_SidPtr , num_sids ) ;
2010-09-07 14:58:19 +02:00
if ( sid_array . sids = = NULL ) {
2008-02-18 10:03:19 +01:00
return NT_STATUS_NO_MEMORY ;
}
2007-05-22 11:30:35 +00:00
2008-02-18 10:03:19 +01:00
for ( i = 0 ; i < num_sids ; i + + ) {
2010-08-26 17:21:39 +02:00
sid_array . sids [ i ] . sid = dom_sid_dup ( mem_ctx , & sids [ i ] ) ;
2008-02-18 10:03:19 +01:00
if ( ! sid_array . sids [ i ] . sid ) {
return NT_STATUS_NO_MEMORY ;
}
}
2007-05-22 11:30:35 +00:00
2009-09-13 00:28:49 +02:00
if ( use_lookupsids3 ) {
struct lsa_TransNameArray2 lsa_names2 ;
uint32_t n ;
2009-09-17 09:42:49 +02:00
ZERO_STRUCT ( lsa_names2 ) ;
2010-09-07 14:58:19 +02:00
status = dcerpc_lsa_LookupSids3 ( h ,
mem_ctx ,
2009-09-13 00:28:49 +02:00
& sid_array ,
& ref_domains ,
& lsa_names2 ,
level ,
& count ,
2011-01-05 17:22:18 +01:00
LSA_LOOKUP_OPTION_SEARCH_ISOLATED_NAMES ,
2011-01-05 17:24:51 +01:00
LSA_CLIENT_REVISION_2 ,
2010-09-07 14:58:19 +02:00
& result ) ;
2011-01-17 13:40:12 +01:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2010-09-07 14:58:19 +02:00
return status ;
}
2009-09-13 00:28:49 +02:00
2010-09-07 14:58:19 +02:00
if ( ! NT_STATUS_IS_ERR ( result ) ) {
lsa_names . count = lsa_names2 . count ;
lsa_names . names = talloc_array ( mem_ctx ,
struct lsa_TranslatedName ,
lsa_names . count ) ;
if ( lsa_names . names = = NULL ) {
2009-09-13 00:28:49 +02:00
return NT_STATUS_NO_MEMORY ;
}
for ( n = 0 ; n < lsa_names . count ; n + + ) {
lsa_names . names [ n ] . sid_type = lsa_names2 . names [ n ] . sid_type ;
lsa_names . names [ n ] . name = lsa_names2 . names [ n ] . name ;
lsa_names . names [ n ] . sid_index = lsa_names2 . names [ n ] . sid_index ;
}
}
} else {
2010-09-07 14:58:19 +02:00
status = dcerpc_lsa_LookupSids ( h ,
mem_ctx ,
2009-09-13 00:28:49 +02:00
pol ,
& sid_array ,
& ref_domains ,
& lsa_names ,
level ,
2010-09-07 14:58:19 +02:00
& count ,
& result ) ;
2009-09-13 00:28:49 +02:00
}
2007-05-22 11:30:35 +00:00
2010-09-07 14:58:19 +02:00
DEBUG ( 10 , ( " LSA_LOOKUPSIDS returned status: '%s', result: '%s', "
" mapped count = %d' \n " ,
nt_errstr ( status ) , nt_errstr ( result ) , count ) ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
2007-09-19 12:05:47 +00:00
2008-02-18 10:03:19 +01:00
if ( ! NT_STATUS_IS_OK ( result ) & &
! NT_STATUS_EQUAL ( result , NT_STATUS_NONE_MAPPED ) & &
! NT_STATUS_EQUAL ( result , STATUS_SOME_UNMAPPED ) )
2007-05-22 11:30:35 +00:00
{
2010-09-07 14:58:19 +02:00
* presult = result ;
return status ;
2007-05-22 11:30:35 +00:00
}
/* Return output parameters */
2008-02-18 10:03:19 +01:00
if ( NT_STATUS_EQUAL ( result , NT_STATUS_NONE_MAPPED ) | |
( count = = 0 ) )
2007-09-19 12:03:02 +00:00
{
2007-08-29 14:34:15 +00:00
for ( i = 0 ; i < num_sids ; i + + ) {
( names ) [ i ] = NULL ;
( domains ) [ i ] = NULL ;
( types ) [ i ] = SID_NAME_UNKNOWN ;
}
2010-09-07 14:58:19 +02:00
* presult = NT_STATUS_NONE_MAPPED ;
return status ;
2007-05-22 11:30:35 +00:00
}
for ( i = 0 ; i < num_sids ; i + + ) {
2008-02-18 10:03:19 +01:00
const char * name , * dom_name ;
uint32_t dom_idx = lsa_names . names [ i ] . sid_index ;
2007-05-22 11:30:35 +00:00
/* Translate optimised name through domain index array */
if ( dom_idx ! = 0xffffffff ) {
2008-02-18 10:03:19 +01:00
dom_name = ref_domains - > domains [ dom_idx ] . name . string ;
name = lsa_names . names [ i ] . name . string ;
2007-05-22 11:30:35 +00:00
2008-04-03 00:23:50 +02:00
if ( name ) {
2010-06-11 11:48:37 +02:00
( names ) [ i ] = talloc_strdup ( names , name ) ;
2008-04-03 00:23:50 +02:00
if ( ( names ) [ i ] = = NULL ) {
DEBUG ( 0 , ( " cli_lsa_lookup_sids_noalloc(): out of memory \n " ) ) ;
2010-09-07 14:58:19 +02:00
* presult = NT_STATUS_UNSUCCESSFUL ;
return status ;
2008-04-03 00:23:50 +02:00
}
} else {
( names ) [ i ] = NULL ;
}
2010-06-11 11:48:37 +02:00
domains [ i ] = talloc_strdup ( domains ,
dom_name ? dom_name : " " ) ;
2008-02-18 10:03:19 +01:00
( types ) [ i ] = lsa_names . names [ i ] . sid_type ;
2008-04-03 00:23:50 +02:00
if ( ( ( domains ) [ i ] = = NULL ) ) {
2007-06-09 00:13:07 +00:00
DEBUG ( 0 , ( " cli_lsa_lookup_sids_noalloc(): out of memory \n " ) ) ;
2010-09-07 14:58:19 +02:00
* presult = NT_STATUS_UNSUCCESSFUL ;
return status ;
2007-05-22 11:30:35 +00:00
}
} else {
2007-05-22 21:17:31 +00:00
( names ) [ i ] = NULL ;
( domains ) [ i ] = NULL ;
2007-05-22 11:30:35 +00:00
( types ) [ i ] = SID_NAME_UNKNOWN ;
}
}
2010-09-07 14:58:19 +02:00
* presult = NT_STATUS_OK ;
return status ;
2007-05-22 11:30:35 +00:00
}
2007-09-19 12:45:35 +00:00
/* Lookup a list of sids
2007-05-22 11:30:35 +00:00
*
* do it the right way : there is a limit ( of 20480 for w2k3 ) entries
* returned by this call . when the sids list contains more entries ,
* empty lists are returned . This version of lsa_lookup_sids passes
* the list of sids in hunks of LOOKUP_SIDS_HUNK_SIZE to the lsa call . */
/* This constant defines the limit of how many sids to look up
* in one call ( maximum ) . the limit from the server side is
* at 20480 for win2k3 , but we keep it at a save 1000 for now . */
# define LOOKUP_SIDS_HUNK_SIZE 1000
2010-09-07 14:58:19 +02:00
static NTSTATUS dcerpc_lsa_lookup_sids_generic ( struct dcerpc_binding_handle * h ,
2009-09-13 00:28:49 +02:00
TALLOC_CTX * mem_ctx ,
struct policy_handle * pol ,
int num_sids ,
2010-05-21 11:25:01 +10:00
const struct dom_sid * sids ,
2009-09-13 00:28:49 +02:00
char * * * pdomains ,
char * * * pnames ,
enum lsa_SidType * * ptypes ,
2010-09-07 14:58:19 +02:00
bool use_lookupsids3 ,
NTSTATUS * presult )
2007-05-22 11:30:35 +00:00
{
2011-02-09 09:46:43 +01:00
NTSTATUS status = NT_STATUS_OK ;
2007-05-22 11:30:35 +00:00
NTSTATUS result = NT_STATUS_OK ;
int sids_left = 0 ;
int sids_processed = 0 ;
2010-05-21 11:25:01 +10:00
const struct dom_sid * hunk_sids = sids ;
2008-03-05 16:20:34 +01:00
char * * hunk_domains ;
char * * hunk_names ;
enum lsa_SidType * hunk_types ;
char * * domains = NULL ;
char * * names = NULL ;
enum lsa_SidType * types = NULL ;
2010-11-23 14:28:45 +01:00
bool have_mapped = false ;
bool have_unmapped = false ;
2007-05-22 11:30:35 +00:00
if ( num_sids ) {
2011-06-07 11:30:12 +10:00
if ( ! ( domains = talloc_array ( mem_ctx , char * , num_sids ) ) ) {
2007-09-19 22:11:09 +00:00
DEBUG ( 0 , ( " rpccli_lsa_lookup_sids(): out of memory \n " ) ) ;
2011-01-17 13:39:21 +01:00
status = NT_STATUS_NO_MEMORY ;
2007-09-19 11:01:44 +00:00
goto fail ;
2007-05-22 11:30:35 +00:00
}
2011-06-07 11:30:12 +10:00
if ( ! ( names = talloc_array ( mem_ctx , char * , num_sids ) ) ) {
2007-09-19 22:11:09 +00:00
DEBUG ( 0 , ( " rpccli_lsa_lookup_sids(): out of memory \n " ) ) ;
2011-01-17 13:39:21 +01:00
status = NT_STATUS_NO_MEMORY ;
2007-09-19 11:01:44 +00:00
goto fail ;
2007-05-22 11:30:35 +00:00
}
2011-06-07 11:30:12 +10:00
if ( ! ( types = talloc_array ( mem_ctx , enum lsa_SidType , num_sids ) ) ) {
2007-09-19 22:11:09 +00:00
DEBUG ( 0 , ( " rpccli_lsa_lookup_sids(): out of memory \n " ) ) ;
2011-01-17 13:39:21 +01:00
status = NT_STATUS_NO_MEMORY ;
2007-09-19 11:01:44 +00:00
goto fail ;
2007-05-22 11:30:35 +00:00
}
}
2007-09-19 12:45:35 +00:00
2007-05-22 11:30:35 +00:00
sids_left = num_sids ;
2008-03-05 16:20:34 +01:00
hunk_domains = domains ;
hunk_names = names ;
hunk_types = types ;
2007-05-22 11:30:35 +00:00
while ( sids_left > 0 ) {
int hunk_num_sids ;
2011-01-17 13:39:21 +01:00
NTSTATUS hunk_result = NT_STATUS_UNSUCCESSFUL ;
2007-05-22 11:30:35 +00:00
2007-09-19 12:45:35 +00:00
hunk_num_sids = ( ( sids_left > LOOKUP_SIDS_HUNK_SIZE )
? LOOKUP_SIDS_HUNK_SIZE
2007-05-22 11:30:35 +00:00
: sids_left ) ;
2007-09-19 22:11:09 +00:00
DEBUG ( 10 , ( " rpccli_lsa_lookup_sids: processing items "
2007-09-19 12:45:35 +00:00
" %d -- %d of %d. \n " ,
sids_processed ,
2007-05-22 11:30:35 +00:00
sids_processed + hunk_num_sids - 1 ,
num_sids ) ) ;
2010-09-07 14:58:19 +02:00
status = dcerpc_lsa_lookup_sids_noalloc ( h ,
mem_ctx ,
pol ,
hunk_num_sids ,
hunk_sids ,
hunk_domains ,
hunk_names ,
hunk_types ,
use_lookupsids3 ,
& hunk_result ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
goto fail ;
}
2007-05-22 11:30:35 +00:00
if ( ! NT_STATUS_IS_OK ( hunk_result ) & &
! NT_STATUS_EQUAL ( hunk_result , STATUS_SOME_UNMAPPED ) & &
2007-09-19 12:45:35 +00:00
! NT_STATUS_EQUAL ( hunk_result , NT_STATUS_NONE_MAPPED ) )
2007-05-22 11:30:35 +00:00
{
/* An actual error occured */
2010-09-07 14:58:19 +02:00
* presult = hunk_result ;
2007-09-19 11:01:44 +00:00
goto fail ;
2007-05-22 11:30:35 +00:00
}
2010-11-23 14:28:45 +01:00
if ( NT_STATUS_IS_OK ( hunk_result ) ) {
have_mapped = true ;
}
if ( NT_STATUS_EQUAL ( hunk_result , NT_STATUS_NONE_MAPPED ) ) {
have_unmapped = true ;
}
if ( NT_STATUS_EQUAL ( hunk_result , STATUS_SOME_UNMAPPED ) ) {
int i ;
for ( i = 0 ; i < hunk_num_sids ; i + + ) {
if ( hunk_types [ i ] = = SID_NAME_UNKNOWN ) {
have_unmapped = true ;
} else {
have_mapped = true ;
}
}
2007-05-22 11:30:35 +00:00
}
sids_left - = hunk_num_sids ;
sids_processed + = hunk_num_sids ; /* only used in DEBUG */
hunk_sids + = hunk_num_sids ;
hunk_domains + = hunk_num_sids ;
hunk_names + = hunk_num_sids ;
hunk_types + = hunk_num_sids ;
}
2008-03-05 16:20:34 +01:00
* pdomains = domains ;
* pnames = names ;
* ptypes = types ;
2010-11-23 14:28:45 +01:00
if ( ! have_mapped ) {
2010-09-07 14:58:19 +02:00
result = NT_STATUS_NONE_MAPPED ;
2010-11-23 14:28:45 +01:00
}
if ( have_unmapped ) {
2010-09-07 14:58:19 +02:00
result = STATUS_SOME_UNMAPPED ;
2010-11-23 14:28:45 +01:00
}
2010-09-07 14:58:19 +02:00
* presult = result ;
return status ;
2007-09-19 11:01:44 +00:00
fail :
2008-03-05 16:20:34 +01:00
TALLOC_FREE ( domains ) ;
TALLOC_FREE ( names ) ;
TALLOC_FREE ( types ) ;
2010-09-07 14:58:19 +02:00
return status ;
}
NTSTATUS dcerpc_lsa_lookup_sids ( struct dcerpc_binding_handle * h ,
TALLOC_CTX * mem_ctx ,
struct policy_handle * pol ,
int num_sids ,
const struct dom_sid * sids ,
char * * * pdomains ,
char * * * pnames ,
enum lsa_SidType * * ptypes ,
NTSTATUS * result )
{
return dcerpc_lsa_lookup_sids_generic ( h ,
mem_ctx ,
pol ,
num_sids ,
sids ,
pdomains ,
pnames ,
ptypes ,
false ,
result ) ;
2007-05-22 11:30:35 +00:00
}
2009-09-13 00:28:49 +02:00
NTSTATUS rpccli_lsa_lookup_sids ( struct rpc_pipe_client * cli ,
TALLOC_CTX * mem_ctx ,
struct policy_handle * pol ,
int num_sids ,
2010-05-21 11:25:01 +10:00
const struct dom_sid * sids ,
2009-09-13 00:28:49 +02:00
char * * * pdomains ,
char * * * pnames ,
enum lsa_SidType * * ptypes )
{
2010-09-07 14:58:19 +02:00
NTSTATUS status ;
NTSTATUS result = NT_STATUS_UNSUCCESSFUL ;
status = dcerpc_lsa_lookup_sids_generic ( cli - > binding_handle ,
mem_ctx ,
pol ,
num_sids ,
sids ,
pdomains ,
pnames ,
ptypes ,
false ,
& result ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
return result ;
}
NTSTATUS dcerpc_lsa_lookup_sids3 ( struct dcerpc_binding_handle * h ,
TALLOC_CTX * mem_ctx ,
struct policy_handle * pol ,
int num_sids ,
const struct dom_sid * sids ,
char * * * pdomains ,
char * * * pnames ,
enum lsa_SidType * * ptypes ,
NTSTATUS * result )
{
return dcerpc_lsa_lookup_sids_generic ( h ,
mem_ctx ,
pol ,
num_sids ,
sids ,
pdomains ,
pnames ,
ptypes ,
true ,
result ) ;
2009-09-13 00:28:49 +02:00
}
NTSTATUS rpccli_lsa_lookup_sids3 ( struct rpc_pipe_client * cli ,
TALLOC_CTX * mem_ctx ,
struct policy_handle * pol ,
int num_sids ,
2010-05-21 11:25:01 +10:00
const struct dom_sid * sids ,
2009-09-13 00:28:49 +02:00
char * * * pdomains ,
char * * * pnames ,
enum lsa_SidType * * ptypes )
{
2010-09-07 14:58:19 +02:00
NTSTATUS status ;
NTSTATUS result = NT_STATUS_UNSUCCESSFUL ;
status = dcerpc_lsa_lookup_sids_generic ( cli - > binding_handle ,
mem_ctx ,
pol ,
num_sids ,
sids ,
pdomains ,
pnames ,
ptypes ,
true ,
& result ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
return result ;
2009-09-13 00:28:49 +02:00
}
2001-11-15 06:03:22 +00:00
/** Lookup a list of names */
2000-12-13 12:52:21 +00:00
2010-09-07 15:49:00 +02:00
static NTSTATUS dcerpc_lsa_lookup_names_generic ( struct dcerpc_binding_handle * h ,
2009-09-11 19:35:14 +02:00
TALLOC_CTX * mem_ctx ,
2010-09-07 15:49:00 +02:00
struct policy_handle * pol ,
uint32_t num_names ,
2009-09-11 19:35:14 +02:00
const char * * names ,
const char * * * dom_names ,
2010-09-07 15:49:00 +02:00
enum lsa_LookupNamesLevel level ,
2010-05-21 11:25:01 +10:00
struct dom_sid * * sids ,
2009-09-11 19:35:14 +02:00
enum lsa_SidType * * types ,
2010-09-07 15:49:00 +02:00
bool use_lookupnames4 ,
NTSTATUS * presult )
2000-12-13 12:52:21 +00:00
{
2010-09-07 15:49:00 +02:00
NTSTATUS status ;
2008-02-18 04:30:57 +01:00
struct lsa_String * lsa_names = NULL ;
struct lsa_RefDomainList * domains = NULL ;
struct lsa_TransSidArray sid_array ;
2009-09-11 19:35:14 +02:00
struct lsa_TransSidArray3 sid_array3 ;
2008-02-18 04:30:57 +01:00
uint32_t count = 0 ;
2010-09-07 15:49:00 +02:00
uint32_t i ;
2007-09-19 12:45:35 +00:00
2008-02-18 04:30:57 +01:00
ZERO_STRUCT ( sid_array ) ;
2009-09-11 19:35:14 +02:00
ZERO_STRUCT ( sid_array3 ) ;
2000-12-13 12:52:21 +00:00
2011-06-07 11:30:12 +10:00
lsa_names = talloc_array ( mem_ctx , struct lsa_String , num_names ) ;
2010-09-07 15:49:00 +02:00
if ( lsa_names = = NULL ) {
2008-02-18 04:30:57 +01:00
return NT_STATUS_NO_MEMORY ;
}
2005-09-30 17:13:37 +00:00
2010-09-07 15:49:00 +02:00
for ( i = 0 ; i < num_names ; i + + ) {
2008-02-18 04:30:57 +01:00
init_lsa_String ( & lsa_names [ i ] , names [ i ] ) ;
}
2000-12-13 12:52:21 +00:00
2009-09-11 19:35:14 +02:00
if ( use_lookupnames4 ) {
2010-09-07 15:49:00 +02:00
status = dcerpc_lsa_LookupNames4 ( h ,
mem_ctx ,
2009-09-11 19:35:14 +02:00
num_names ,
lsa_names ,
& domains ,
& sid_array3 ,
level ,
& count ,
2011-01-05 17:34:30 +01:00
LSA_LOOKUP_OPTION_SEARCH_ISOLATED_NAMES ,
2011-01-05 17:35:36 +01:00
LSA_CLIENT_REVISION_2 ,
2010-09-07 15:49:00 +02:00
presult ) ;
2009-09-11 19:35:14 +02:00
} else {
2010-09-07 15:49:00 +02:00
status = dcerpc_lsa_LookupNames ( h ,
mem_ctx ,
pol ,
2009-09-11 19:35:14 +02:00
num_names ,
lsa_names ,
& domains ,
& sid_array ,
level ,
2010-09-07 15:49:00 +02:00
& count ,
presult ) ;
}
if ( ! NT_STATUS_IS_OK ( status ) ) {
goto done ;
2009-09-11 19:35:14 +02:00
}
2000-12-13 12:52:21 +00:00
2010-09-07 15:49:00 +02:00
if ( ! NT_STATUS_IS_OK ( * presult ) & &
! NT_STATUS_EQUAL ( * presult , STATUS_SOME_UNMAPPED ) ) {
2000-12-13 12:52:21 +00:00
/* An actual error occured */
goto done ;
}
/* Return output parameters */
2008-02-18 04:30:57 +01:00
if ( count = = 0 ) {
2010-09-07 15:49:00 +02:00
* presult = NT_STATUS_NONE_MAPPED ;
2001-11-22 08:31:50 +00:00
goto done ;
}
2007-04-30 02:39:34 +00:00
if ( num_names ) {
2011-06-07 11:30:12 +10:00
if ( ! ( ( * sids = talloc_array ( mem_ctx , struct dom_sid , num_names ) ) ) ) {
2007-04-30 02:39:34 +00:00
DEBUG ( 0 , ( " cli_lsa_lookup_sids(): out of memory \n " ) ) ;
2010-09-07 15:49:00 +02:00
* presult = NT_STATUS_NO_MEMORY ;
2007-04-30 02:39:34 +00:00
goto done ;
}
2000-12-13 12:52:21 +00:00
2011-06-07 11:30:12 +10:00
if ( ! ( ( * types = talloc_array ( mem_ctx , enum lsa_SidType , num_names ) ) ) ) {
2006-02-03 22:19:41 +00:00
DEBUG ( 0 , ( " cli_lsa_lookup_sids(): out of memory \n " ) ) ;
2010-09-07 15:49:00 +02:00
* presult = NT_STATUS_NO_MEMORY ;
2006-02-03 22:19:41 +00:00
goto done ;
}
2007-04-30 02:39:34 +00:00
if ( dom_names ! = NULL ) {
2011-06-07 11:30:12 +10:00
* dom_names = talloc_array ( mem_ctx , const char * , num_names ) ;
2007-04-30 02:39:34 +00:00
if ( * dom_names = = NULL ) {
DEBUG ( 0 , ( " cli_lsa_lookup_sids(): out of memory \n " ) ) ;
2010-09-07 15:49:00 +02:00
* presult = NT_STATUS_NO_MEMORY ;
2007-04-30 02:39:34 +00:00
goto done ;
}
}
} else {
* sids = NULL ;
* types = NULL ;
if ( dom_names ! = NULL ) {
* dom_names = NULL ;
}
2006-02-03 22:19:41 +00:00
}
2002-04-14 11:21:25 +00:00
for ( i = 0 ; i < num_names ; i + + ) {
2009-09-11 19:35:14 +02:00
uint32_t dom_idx ;
2010-05-21 11:25:01 +10:00
struct dom_sid * sid = & ( * sids ) [ i ] ;
2000-12-13 12:52:21 +00:00
2009-09-11 19:35:14 +02:00
if ( use_lookupnames4 ) {
dom_idx = sid_array3 . sids [ i ] . sid_index ;
( * types ) [ i ] = sid_array3 . sids [ i ] . sid_type ;
} else {
dom_idx = sid_array . sids [ i ] . sid_index ;
( * types ) [ i ] = sid_array . sids [ i ] . sid_type ;
}
2000-12-13 12:52:21 +00:00
/* Translate optimised sid through domain index array */
2006-02-03 22:19:41 +00:00
if ( dom_idx = = 0xffffffff ) {
/* Nothing to do, this is unknown */
ZERO_STRUCTP ( sid ) ;
( * types ) [ i ] = SID_NAME_UNKNOWN ;
continue ;
}
2000-12-13 12:52:21 +00:00
2009-09-11 19:35:14 +02:00
if ( use_lookupnames4 ) {
sid_copy ( sid , sid_array3 . sids [ i ] . sid ) ;
} else {
sid_copy ( sid , domains - > domains [ dom_idx ] . sid ) ;
2000-12-13 12:52:21 +00:00
2009-09-11 19:35:14 +02:00
if ( sid_array . sids [ i ] . rid ! = 0xffffffff ) {
sid_append_rid ( sid , sid_array . sids [ i ] . rid ) ;
}
2006-02-03 22:19:41 +00:00
}
2000-12-13 12:52:21 +00:00
2006-02-03 22:19:41 +00:00
if ( dom_names = = NULL ) {
continue ;
2000-12-13 12:52:21 +00:00
}
2006-02-03 22:19:41 +00:00
2008-02-18 04:30:57 +01:00
( * dom_names ) [ i ] = domains - > domains [ dom_idx ] . name . string ;
2000-12-13 12:52:21 +00:00
}
done :
2010-09-07 15:49:00 +02:00
return status ;
}
2000-12-15 01:02:11 +00:00
2010-09-07 15:49:00 +02:00
NTSTATUS dcerpc_lsa_lookup_names ( struct dcerpc_binding_handle * h ,
TALLOC_CTX * mem_ctx ,
struct policy_handle * pol ,
uint32_t num_names ,
const char * * names ,
const char * * * dom_names ,
enum lsa_LookupNamesLevel level ,
struct dom_sid * * sids ,
enum lsa_SidType * * types ,
NTSTATUS * result )
{
return dcerpc_lsa_lookup_names_generic ( h ,
mem_ctx ,
pol ,
num_names ,
names ,
dom_names ,
level ,
sids ,
types ,
false ,
result ) ;
2002-08-12 13:40:59 +00:00
}
2009-09-11 19:35:14 +02:00
NTSTATUS rpccli_lsa_lookup_names ( struct rpc_pipe_client * cli ,
TALLOC_CTX * mem_ctx ,
2010-09-07 15:49:00 +02:00
struct policy_handle * pol ,
int num_names ,
2009-09-11 19:35:14 +02:00
const char * * names ,
const char * * * dom_names ,
int level ,
2010-05-21 11:25:01 +10:00
struct dom_sid * * sids ,
2009-09-11 19:35:14 +02:00
enum lsa_SidType * * types )
{
2010-09-07 15:49:00 +02:00
NTSTATUS status ;
NTSTATUS result = NT_STATUS_UNSUCCESSFUL ;
status = dcerpc_lsa_lookup_names ( cli - > binding_handle ,
mem_ctx ,
pol ,
num_names ,
names ,
dom_names ,
level ,
sids ,
types ,
& result ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
return result ;
}
NTSTATUS dcerpc_lsa_lookup_names4 ( struct dcerpc_binding_handle * h ,
TALLOC_CTX * mem_ctx ,
struct policy_handle * pol ,
uint32_t num_names ,
const char * * names ,
const char * * * dom_names ,
enum lsa_LookupNamesLevel level ,
struct dom_sid * * sids ,
enum lsa_SidType * * types ,
NTSTATUS * result )
{
return dcerpc_lsa_lookup_names_generic ( h ,
mem_ctx ,
pol ,
num_names ,
names ,
dom_names ,
level ,
sids ,
types ,
true ,
result ) ;
2009-09-11 19:35:14 +02:00
}
NTSTATUS rpccli_lsa_lookup_names4 ( struct rpc_pipe_client * cli ,
TALLOC_CTX * mem_ctx ,
2010-09-07 15:49:00 +02:00
struct policy_handle * pol ,
int num_names ,
2009-09-11 19:35:14 +02:00
const char * * names ,
const char * * * dom_names ,
int level ,
2010-05-21 11:25:01 +10:00
struct dom_sid * * sids ,
2009-09-11 19:35:14 +02:00
enum lsa_SidType * * types )
{
2010-09-07 15:49:00 +02:00
NTSTATUS status ;
NTSTATUS result = NT_STATUS_UNSUCCESSFUL ;
status = dcerpc_lsa_lookup_names4 ( cli - > binding_handle ,
mem_ctx ,
pol ,
num_names ,
names ,
dom_names ,
level ,
sids ,
types ,
& result ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
return result ;
2009-09-11 19:35:14 +02:00
}