2009-03-16 07:20:28 +03:00
/*
2016-04-20 08:10:41 +03:00
Unix SMB / CIFS implementation .
2009-03-16 07:20:28 +03:00
Helper functions for applying replicated objects
Copyright ( C ) Stefan Metzmacher < metze @ samba . org > 2007
2009-09-10 09:50:32 +04:00
Copyright ( C ) Andrew Bartlett < abartlet @ samba . org > 2009
2009-03-16 07:20:28 +03: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 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"
# include "../lib/util/dlinklist.h"
# include "librpc/gen_ndr/ndr_misc.h"
# include "librpc/gen_ndr/ndr_drsuapi.h"
# include "librpc/gen_ndr/ndr_drsblobs.h"
2018-10-09 12:40:17 +03:00
# include "../lib/crypto/arcfour.h"
# include "zlib.h"
2009-03-16 07:20:28 +03:00
# include "../libcli/drsuapi/drsuapi.h"
# include "libcli/auth/libcli_auth.h"
2015-08-19 04:26:41 +03:00
# include "dsdb/samdb/samdb.h"
2009-03-16 07:20:28 +03:00
2019-05-15 09:07:32 +03:00
# include <gnutls/gnutls.h>
# include <gnutls/crypto.h>
2009-03-16 07:20:28 +03:00
WERROR drsuapi_decrypt_attribute_value ( TALLOC_CTX * mem_ctx ,
const DATA_BLOB * gensec_skey ,
bool rid_crypt ,
uint32_t rid ,
DATA_BLOB * in ,
DATA_BLOB * out )
{
DATA_BLOB confounder ;
DATA_BLOB enc_buffer ;
2019-05-15 09:07:32 +03:00
gnutls_hash_hd_t hash_hnd = NULL ;
2009-03-16 07:20:28 +03:00
uint8_t _enc_key [ 16 ] ;
DATA_BLOB enc_key ;
DATA_BLOB dec_buffer ;
uint32_t crc32_given ;
uint32_t crc32_calc ;
DATA_BLOB checked_buffer ;
DATA_BLOB plain_buffer ;
2019-05-15 09:07:32 +03:00
WERROR result ;
int rc ;
2009-03-16 07:20:28 +03:00
/*
* users with rid = = 0 should not exist
*/
if ( rid_crypt & & rid = = 0 ) {
return WERR_DS_DRA_INVALID_PARAMETER ;
}
/*
* the first 16 bytes at the beginning are the confounder
* followed by the 4 byte crc32 checksum
*/
if ( in - > length < 20 ) {
return WERR_DS_DRA_INVALID_PARAMETER ;
}
confounder = data_blob_const ( in - > data , 16 ) ;
enc_buffer = data_blob_const ( in - > data + 16 , in - > length - 16 ) ;
/*
* build the encryption key md5 over the session key followed
* by the confounder
*
* here the gensec session key is used and
* not the dcerpc ncacn_ip_tcp " SystemLibraryDTC " key !
*/
enc_key = data_blob_const ( _enc_key , sizeof ( _enc_key ) ) ;
2019-05-15 09:07:32 +03:00
rc = gnutls_hash_init ( & hash_hnd , GNUTLS_DIG_MD5 ) ;
if ( rc < 0 ) {
result = WERR_NOT_ENOUGH_MEMORY ;
goto out ;
}
rc = gnutls_hash ( hash_hnd , gensec_skey - > data , gensec_skey - > length ) ;
if ( rc < 0 ) {
gnutls_hash_deinit ( hash_hnd , NULL ) ;
result = WERR_INTERNAL_ERROR ;
goto out ;
}
rc = gnutls_hash ( hash_hnd , confounder . data , confounder . length ) ;
if ( rc < 0 ) {
gnutls_hash_deinit ( hash_hnd , NULL ) ;
result = WERR_INTERNAL_ERROR ;
goto out ;
}
gnutls_hash_deinit ( hash_hnd , enc_key . data ) ;
2009-03-16 07:20:28 +03:00
/*
* copy the encrypted buffer part and
* decrypt it using the created encryption key using arcfour
*/
dec_buffer = data_blob_const ( enc_buffer . data , enc_buffer . length ) ;
arcfour_crypt_blob ( dec_buffer . data , dec_buffer . length , & enc_key ) ;
2019-05-15 09:07:32 +03:00
ZERO_ARRAY_LEN ( enc_key . data , enc_key . length ) ;
2009-03-16 07:20:28 +03:00
/*
* the first 4 byte are the crc32 checksum
* of the remaining bytes
*/
crc32_given = IVAL ( dec_buffer . data , 0 ) ;
2018-10-09 12:40:17 +03:00
crc32_calc = crc32 ( 0 , Z_NULL , 0 ) ;
crc32_calc = crc32 ( crc32_calc ,
dec_buffer . data + 4 ,
dec_buffer . length - 4 ) ;
2009-03-16 07:20:28 +03:00
checked_buffer = data_blob_const ( dec_buffer . data + 4 , dec_buffer . length - 4 ) ;
plain_buffer = data_blob_talloc ( mem_ctx , checked_buffer . data , checked_buffer . length ) ;
W_ERROR_HAVE_NO_MEMORY ( plain_buffer . data ) ;
2009-09-10 11:42:13 +04:00
if ( crc32_given ! = crc32_calc ) {
2019-05-15 09:07:32 +03:00
result = W_ERROR ( HRES_ERROR_V ( HRES_SEC_E_DECRYPT_FAILURE ) ) ;
goto out ;
2009-09-10 11:42:13 +04:00
}
2009-03-16 07:20:28 +03:00
/*
* The following rid_crypt obfuscation isn ' t session specific
* and not really needed here , because we allways know the rid of the
* user account .
*
* some attributes with this ' additional encryption ' include
* dBCSPwd , unicodePwd , ntPwdHistory , lmPwdHistory
*
* But for the rest of samba it ' s easier when we remove this static
* obfuscation here
*/
if ( rid_crypt ) {
uint32_t i , num_hashes ;
if ( ( checked_buffer . length % 16 ) ! = 0 ) {
2019-05-15 09:07:32 +03:00
result = WERR_DS_DRA_INVALID_PARAMETER ;
goto out ;
2009-03-16 07:20:28 +03:00
}
num_hashes = plain_buffer . length / 16 ;
for ( i = 0 ; i < num_hashes ; i + + ) {
uint32_t offset = i * 16 ;
sam_rid_crypt ( rid , checked_buffer . data + offset , plain_buffer . data + offset , 0 ) ;
}
}
* out = plain_buffer ;
2019-05-15 09:07:32 +03:00
result = WERR_OK ;
out :
return result ;
2009-03-16 07:20:28 +03:00
}
WERROR drsuapi_decrypt_attribute ( TALLOC_CTX * mem_ctx ,
const DATA_BLOB * gensec_skey ,
uint32_t rid ,
2015-08-19 04:26:41 +03:00
uint32_t dsdb_repl_flags ,
2009-03-16 07:20:28 +03:00
struct drsuapi_DsReplicaAttribute * attr )
{
WERROR status ;
DATA_BLOB * enc_data ;
DATA_BLOB plain_data ;
bool rid_crypt = false ;
if ( attr - > value_ctr . num_values = = 0 ) {
return WERR_OK ;
}
switch ( attr - > attid ) {
2010-10-29 03:22:35 +04:00
case DRSUAPI_ATTID_dBCSPwd :
case DRSUAPI_ATTID_unicodePwd :
case DRSUAPI_ATTID_ntPwdHistory :
case DRSUAPI_ATTID_lmPwdHistory :
2009-03-16 07:20:28 +03:00
rid_crypt = true ;
break ;
2010-10-29 03:22:35 +04:00
case DRSUAPI_ATTID_supplementalCredentials :
case DRSUAPI_ATTID_priorValue :
case DRSUAPI_ATTID_currentValue :
case DRSUAPI_ATTID_trustAuthOutgoing :
case DRSUAPI_ATTID_trustAuthIncoming :
case DRSUAPI_ATTID_initialAuthOutgoing :
case DRSUAPI_ATTID_initialAuthIncoming :
2009-03-16 07:20:28 +03:00
break ;
default :
return WERR_OK ;
}
2015-08-19 04:26:41 +03:00
if ( dsdb_repl_flags & DSDB_REPL_FLAG_EXPECT_NO_SECRETS ) {
return WERR_TOO_MANY_SECRETS ;
}
2009-03-16 07:20:28 +03:00
if ( attr - > value_ctr . num_values > 1 ) {
return WERR_DS_DRA_INVALID_PARAMETER ;
}
if ( ! attr - > value_ctr . values [ 0 ] . blob ) {
return WERR_DS_DRA_INVALID_PARAMETER ;
}
enc_data = attr - > value_ctr . values [ 0 ] . blob ;
status = drsuapi_decrypt_attribute_value ( mem_ctx ,
gensec_skey ,
rid_crypt ,
rid ,
enc_data ,
& plain_data ) ;
W_ERROR_NOT_OK_RETURN ( status ) ;
talloc_free ( attr - > value_ctr . values [ 0 ] . blob - > data ) ;
* attr - > value_ctr . values [ 0 ] . blob = plain_data ;
return WERR_OK ;
}
2009-09-10 09:50:32 +04:00
2009-09-10 11:42:13 +04:00
static WERROR drsuapi_encrypt_attribute_value ( TALLOC_CTX * mem_ctx ,
const DATA_BLOB * gensec_skey ,
bool rid_crypt ,
uint32_t rid ,
DATA_BLOB * in ,
DATA_BLOB * out )
2009-09-10 09:50:32 +04:00
{
DATA_BLOB rid_crypt_out = data_blob ( NULL , 0 ) ;
DATA_BLOB confounder ;
2019-05-15 09:08:15 +03:00
gnutls_hash_hd_t hash_hnd = NULL ;
2009-09-10 09:50:32 +04:00
uint8_t _enc_key [ 16 ] ;
DATA_BLOB enc_key ;
DATA_BLOB enc_buffer ;
uint32_t crc32_calc ;
2019-05-15 09:08:15 +03:00
WERROR result ;
int rc ;
2009-09-10 09:50:32 +04:00
/*
* users with rid = = 0 should not exist
*/
if ( rid_crypt & & rid = = 0 ) {
return WERR_DS_DRA_INVALID_PARAMETER ;
}
/*
* The following rid_crypt obfuscation isn ' t session specific
* and not really needed here , because we allways know the rid of the
* user account .
*
* some attributes with this ' additional encryption ' include
* dBCSPwd , unicodePwd , ntPwdHistory , lmPwdHistory
*
* But for the rest of samba it ' s easier when we remove this static
* obfuscation here
*/
if ( rid_crypt ) {
uint32_t i , num_hashes ;
rid_crypt_out = data_blob_talloc ( mem_ctx , in - > data , in - > length ) ;
W_ERROR_HAVE_NO_MEMORY ( rid_crypt_out . data ) ;
if ( ( rid_crypt_out . length % 16 ) ! = 0 ) {
return WERR_DS_DRA_INVALID_PARAMETER ;
}
num_hashes = rid_crypt_out . length / 16 ;
for ( i = 0 ; i < num_hashes ; i + + ) {
uint32_t offset = i * 16 ;
sam_rid_crypt ( rid , in - > data + offset , rid_crypt_out . data + offset , 1 ) ;
}
in = & rid_crypt_out ;
}
/*
* the first 16 bytes at the beginning are the confounder
* followed by the 4 byte crc32 checksum
*/
enc_buffer = data_blob_talloc ( mem_ctx , NULL , in - > length + 20 ) ;
if ( ! enc_buffer . data ) {
talloc_free ( rid_crypt_out . data ) ;
2015-12-03 17:24:14 +03:00
return WERR_NOT_ENOUGH_MEMORY ;
2009-09-10 09:50:32 +04:00
} ;
confounder = data_blob_const ( enc_buffer . data , 16 ) ;
generate_random_buffer ( confounder . data , confounder . length ) ;
/*
* build the encryption key md5 over the session key followed
* by the confounder
*
* here the gensec session key is used and
* not the dcerpc ncacn_ip_tcp " SystemLibraryDTC " key !
*/
enc_key = data_blob_const ( _enc_key , sizeof ( _enc_key ) ) ;
2019-05-15 09:08:15 +03:00
rc = gnutls_hash_init ( & hash_hnd , GNUTLS_DIG_MD5 ) ;
if ( rc < 0 ) {
result = WERR_NOT_ENOUGH_MEMORY ;
goto out ;
}
rc = gnutls_hash ( hash_hnd , gensec_skey - > data , gensec_skey - > length ) ;
if ( rc < 0 ) {
gnutls_hash_deinit ( hash_hnd , NULL ) ;
result = WERR_INTERNAL_ERROR ;
goto out ;
}
rc = gnutls_hash ( hash_hnd , confounder . data , confounder . length ) ;
if ( rc < 0 ) {
gnutls_hash_deinit ( hash_hnd , NULL ) ;
result = WERR_INTERNAL_ERROR ;
goto out ;
}
gnutls_hash_deinit ( hash_hnd , enc_key . data ) ;
2009-09-10 09:50:32 +04:00
/*
* the first 4 byte are the crc32 checksum
* of the remaining bytes
*/
2018-10-09 12:40:17 +03:00
crc32_calc = crc32 ( 0 , Z_NULL , 0 ) ;
crc32_calc = crc32 ( crc32_calc , in - > data , in - > length ) ;
2009-09-10 11:42:13 +04:00
SIVAL ( enc_buffer . data , 16 , crc32_calc ) ;
2009-09-10 09:50:32 +04:00
/*
* copy the plain buffer part and
* encrypt it using the created encryption key using arcfour
*/
memcpy ( enc_buffer . data + 20 , in - > data , in - > length ) ;
talloc_free ( rid_crypt_out . data ) ;
2009-09-10 11:42:13 +04:00
arcfour_crypt_blob ( enc_buffer . data + 16 , enc_buffer . length - 16 , & enc_key ) ;
2009-09-10 09:50:32 +04:00
2019-05-15 09:08:15 +03:00
ZERO_ARRAY_LEN ( enc_key . data , enc_key . length ) ;
2009-09-10 11:42:13 +04:00
2019-05-15 09:08:15 +03:00
* out = enc_buffer ;
result = WERR_OK ;
out :
return result ;
2009-09-10 11:42:13 +04:00
}
/*
encrypt a DRSUAPI attribute ready for sending over the wire
Only some attribute types are encrypted
*/
WERROR drsuapi_encrypt_attribute ( TALLOC_CTX * mem_ctx ,
const DATA_BLOB * gensec_skey ,
uint32_t rid ,
struct drsuapi_DsReplicaAttribute * attr )
{
WERROR status ;
DATA_BLOB * plain_data ;
DATA_BLOB enc_data ;
bool rid_crypt = false ;
if ( attr - > value_ctr . num_values = = 0 ) {
return WERR_OK ;
}
switch ( attr - > attid ) {
2010-10-29 03:22:35 +04:00
case DRSUAPI_ATTID_dBCSPwd :
case DRSUAPI_ATTID_unicodePwd :
case DRSUAPI_ATTID_ntPwdHistory :
case DRSUAPI_ATTID_lmPwdHistory :
2009-09-10 11:42:13 +04:00
rid_crypt = true ;
break ;
2010-10-29 03:22:35 +04:00
case DRSUAPI_ATTID_supplementalCredentials :
case DRSUAPI_ATTID_priorValue :
case DRSUAPI_ATTID_currentValue :
case DRSUAPI_ATTID_trustAuthOutgoing :
case DRSUAPI_ATTID_trustAuthIncoming :
case DRSUAPI_ATTID_initialAuthOutgoing :
case DRSUAPI_ATTID_initialAuthIncoming :
2009-09-10 11:42:13 +04:00
break ;
default :
return WERR_OK ;
}
if ( attr - > value_ctr . num_values > 1 ) {
return WERR_DS_DRA_INVALID_PARAMETER ;
}
if ( ! attr - > value_ctr . values [ 0 ] . blob ) {
return WERR_DS_DRA_INVALID_PARAMETER ;
}
plain_data = attr - > value_ctr . values [ 0 ] . blob ;
status = drsuapi_encrypt_attribute_value ( mem_ctx ,
gensec_skey ,
rid_crypt ,
rid ,
plain_data ,
& enc_data ) ;
W_ERROR_NOT_OK_RETURN ( status ) ;
talloc_free ( attr - > value_ctr . values [ 0 ] . blob - > data ) ;
* attr - > value_ctr . values [ 0 ] . blob = enc_data ;
2009-09-10 09:50:32 +04:00
return WERR_OK ;
}