2011-12-26 22:53:34 +04:00
/*
* fs / cifs / smb2transport . c
*
* Copyright ( C ) International Business Machines Corp . , 2002 , 2011
* Etersoft , 2012
* Author ( s ) : Steve French ( sfrench @ us . ibm . com )
* Jeremy Allison ( jra @ samba . org ) 2006
* Pavel Shilovsky ( pshilovsky @ samba . org ) 2012
*
* This library is free software ; you can redistribute it and / or modify
* it under the terms of the GNU Lesser General Public License as published
* by the Free Software Foundation ; either version 2.1 of the License , or
* ( at your option ) any later version .
*
* This library 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 Lesser General Public License for more details .
*
* You should have received a copy of the GNU Lesser General Public License
* along with this library ; if not , write to the Free Software
* Foundation , Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*/
# include <linux/fs.h>
# include <linux/list.h>
# include <linux/wait.h>
# include <linux/net.h>
# include <linux/delay.h>
# include <linux/uaccess.h>
# include <asm/processor.h>
# include <linux/mempool.h>
2012-09-18 16:20:35 -07:00
# include <linux/highmem.h>
2016-11-03 16:47:37 -07:00
# include <crypto/aead.h>
2011-12-26 22:53:34 +04:00
# include "smb2pdu.h"
# include "cifsglob.h"
# include "cifsproto.h"
# include "smb2proto.h"
# include "cifs_debug.h"
# include "smb2status.h"
2012-09-18 16:20:30 -07:00
# include "smb2glob.h"
2013-07-04 10:35:21 -05:00
static int
smb2_crypto_shash_allocate ( struct TCP_Server_Info * server )
{
2018-02-16 19:19:27 +01:00
return cifs_alloc_hash ( " hmac(sha256) " ,
& server - > secmech . hmacsha256 ,
& server - > secmech . sdeschmacsha256 ) ;
2013-07-04 10:35:21 -05:00
}
static int
smb3_crypto_shash_allocate ( struct TCP_Server_Info * server )
{
2018-02-16 19:19:27 +01:00
struct cifs_secmech * p = & server - > secmech ;
2013-07-04 10:35:21 -05:00
int rc ;
2018-02-16 19:19:27 +01:00
rc = cifs_alloc_hash ( " hmac(sha256) " ,
& p - > hmacsha256 ,
& p - > sdeschmacsha256 ) ;
2013-07-04 10:35:21 -05:00
if ( rc )
2018-02-16 19:19:27 +01:00
goto err ;
2013-07-04 10:35:21 -05:00
2018-02-16 19:19:27 +01:00
rc = cifs_alloc_hash ( " cmac(aes) " , & p - > cmacaes , & p - > sdesccmacaes ) ;
if ( rc )
goto err ;
2013-07-04 10:35:21 -05:00
return 0 ;
2018-02-16 19:19:27 +01:00
err :
cifs_free_hash ( & p - > hmacsha256 , & p - > sdeschmacsha256 ) ;
return rc ;
2013-07-04 10:35:21 -05:00
}
2018-02-16 19:19:28 +01:00
int
smb311_crypto_shash_allocate ( struct TCP_Server_Info * server )
{
struct cifs_secmech * p = & server - > secmech ;
int rc = 0 ;
rc = cifs_alloc_hash ( " hmac(sha256) " ,
& p - > hmacsha256 ,
& p - > sdeschmacsha256 ) ;
if ( rc )
return rc ;
rc = cifs_alloc_hash ( " cmac(aes) " , & p - > cmacaes , & p - > sdesccmacaes ) ;
if ( rc )
goto err ;
rc = cifs_alloc_hash ( " sha512 " , & p - > sha512 , & p - > sdescsha512 ) ;
if ( rc )
goto err ;
return 0 ;
err :
cifs_free_hash ( & p - > cmacaes , & p - > sdesccmacaes ) ;
cifs_free_hash ( & p - > hmacsha256 , & p - > sdeschmacsha256 ) ;
return rc ;
}
2019-09-20 06:31:10 +02:00
static
int smb2_get_sign_key ( __u64 ses_id , struct TCP_Server_Info * server , u8 * key )
{
struct cifs_chan * chan ;
struct cifs_ses * ses = NULL ;
int i ;
int rc = 0 ;
spin_lock ( & cifs_tcp_ses_lock ) ;
list_for_each_entry ( server , & cifs_tcp_ses_list , tcp_ses_list ) {
list_for_each_entry ( ses , & server - > smb_ses_list , smb_ses_list ) {
if ( ses - > Suid = = ses_id )
goto found ;
}
}
cifs_server_dbg ( VFS , " %s: Could not find session 0x%llx \n " ,
__func__ , ses_id ) ;
rc = - ENOENT ;
goto out ;
found :
if ( ses - > binding ) {
/*
* If we are in the process of binding a new channel
* to an existing session , use the master connection
* session key
*/
memcpy ( key , ses - > smb3signingkey , SMB3_SIGN_KEY_SIZE ) ;
goto out ;
}
/*
* Otherwise , use the channel key .
*/
for ( i = 0 ; i < ses - > chan_count ; i + + ) {
chan = ses - > chans + i ;
if ( chan - > server = = server ) {
memcpy ( key , chan - > signkey , SMB3_SIGN_KEY_SIZE ) ;
goto out ;
}
}
cifs_dbg ( VFS ,
" %s: Could not find channel signing key for session 0x%llx \n " ,
__func__ , ses_id ) ;
rc = - ENOENT ;
out :
spin_unlock ( & cifs_tcp_ses_lock ) ;
return rc ;
}
2017-03-03 15:41:38 -08:00
static struct cifs_ses *
smb2_find_smb_ses_unlocked ( struct TCP_Server_Info * server , __u64 ses_id )
2013-08-29 08:35:11 -05:00
{
struct cifs_ses * ses ;
list_for_each_entry ( ses , & server - > smb_ses_list , smb_ses_list ) {
2016-11-03 16:47:37 -07:00
if ( ses - > Suid ! = ses_id )
2013-08-29 08:35:11 -05:00
continue ;
return ses ;
}
2017-03-03 15:41:38 -08:00
return NULL ;
}
struct cifs_ses *
smb2_find_smb_ses ( struct TCP_Server_Info * server , __u64 ses_id )
{
struct cifs_ses * ses ;
spin_lock ( & cifs_tcp_ses_lock ) ;
ses = smb2_find_smb_ses_unlocked ( server , ses_id ) ;
2013-08-29 08:35:11 -05:00
spin_unlock ( & cifs_tcp_ses_lock ) ;
2017-03-03 15:41:38 -08:00
return ses ;
}
static struct cifs_tcon *
smb2_find_smb_sess_tcon_unlocked ( struct cifs_ses * ses , __u32 tid )
{
struct cifs_tcon * tcon ;
list_for_each_entry ( tcon , & ses - > tcon_list , tcon_list ) {
if ( tcon - > tid ! = tid )
continue ;
+ + tcon - > tc_count ;
return tcon ;
}
2013-08-29 08:35:11 -05:00
return NULL ;
}
2017-03-03 15:41:38 -08:00
/*
* Obtain tcon corresponding to the tid in the given
* cifs_ses
*/
struct cifs_tcon *
smb2_find_smb_tcon ( struct TCP_Server_Info * server , __u64 ses_id , __u32 tid )
{
struct cifs_ses * ses ;
struct cifs_tcon * tcon ;
spin_lock ( & cifs_tcp_ses_lock ) ;
ses = smb2_find_smb_ses_unlocked ( server , ses_id ) ;
if ( ! ses ) {
spin_unlock ( & cifs_tcp_ses_lock ) ;
return NULL ;
}
tcon = smb2_find_smb_sess_tcon_unlocked ( ses , tid ) ;
spin_unlock ( & cifs_tcp_ses_lock ) ;
return tcon ;
}
2012-12-08 22:08:06 -06:00
int
2012-09-18 16:20:34 -07:00
smb2_calc_signature ( struct smb_rqst * rqst , struct TCP_Server_Info * server )
2012-09-18 16:20:30 -07:00
{
2015-11-12 22:46:49 -05:00
int rc ;
2012-09-18 16:20:30 -07:00
unsigned char smb2_signature [ SMB2_HMACSHA256_SIZE ] ;
unsigned char * sigptr = smb2_signature ;
2012-09-18 16:20:34 -07:00
struct kvec * iov = rqst - > rq_iov ;
2018-06-12 08:00:58 +10:00
struct smb2_sync_hdr * shdr = ( struct smb2_sync_hdr * ) iov [ 0 ] . iov_base ;
2013-08-29 08:35:11 -05:00
struct cifs_ses * ses ;
2018-08-02 16:39:52 +02:00
struct shash_desc * shash ;
2018-06-23 14:52:24 -03:00
struct smb_rqst drqst ;
2013-08-29 08:35:11 -05:00
2016-11-03 16:47:37 -07:00
ses = smb2_find_smb_ses ( server , shdr - > SessionId ) ;
2013-08-29 08:35:11 -05:00
if ( ! ses ) {
2019-08-28 17:15:35 +10:00
cifs_server_dbg ( VFS , " %s: Could not find session \n " , __func__ ) ;
2013-08-29 08:35:11 -05:00
return 0 ;
}
2012-09-18 16:20:30 -07:00
memset ( smb2_signature , 0x0 , SMB2_HMACSHA256_SIZE ) ;
2016-10-24 15:33:04 -07:00
memset ( shdr - > Signature , 0x0 , SMB2_SIGNATURE_SIZE ) ;
2012-09-18 16:20:30 -07:00
2013-07-04 10:35:21 -05:00
rc = smb2_crypto_shash_allocate ( server ) ;
if ( rc ) {
2019-08-28 17:15:35 +10:00
cifs_server_dbg ( VFS , " %s: sha256 alloc failed \n " , __func__ ) ;
2013-07-04 10:35:21 -05:00
return rc ;
}
2012-09-18 16:20:30 -07:00
rc = crypto_shash_setkey ( server - > secmech . hmacsha256 ,
2018-06-23 14:52:24 -03:00
ses - > auth_key . response , SMB2_NTLMV2_SESSKEY_SIZE ) ;
2012-09-18 16:20:30 -07:00
if ( rc ) {
2019-08-28 17:15:35 +10:00
cifs_server_dbg ( VFS , " %s: Could not update with response \n " , __func__ ) ;
2012-09-18 16:20:30 -07:00
return rc ;
}
2018-08-02 16:39:52 +02:00
shash = & server - > secmech . sdeschmacsha256 - > shash ;
2018-06-23 14:52:24 -03:00
rc = crypto_shash_init ( shash ) ;
2012-09-18 16:20:30 -07:00
if ( rc ) {
2019-08-28 17:15:35 +10:00
cifs_server_dbg ( VFS , " %s: Could not init sha256 " , __func__ ) ;
2012-09-18 16:20:30 -07:00
return rc ;
}
2018-06-23 14:52:24 -03:00
/*
* For SMB2 + , __cifs_calc_signature ( ) expects to sign only the actual
* data , that is , iov [ 0 ] should not contain a rfc1002 length .
*
* Sign the rfc1002 length prior to passing the data ( iov [ 1 - N ] ) down to
* __cifs_calc_signature ( ) .
*/
drqst = * rqst ;
if ( drqst . rq_nvec > = 2 & & iov [ 0 ] . iov_len = = 4 ) {
rc = crypto_shash_update ( shash , iov [ 0 ] . iov_base ,
iov [ 0 ] . iov_len ) ;
if ( rc ) {
2019-08-28 17:15:35 +10:00
cifs_server_dbg ( VFS , " %s: Could not update with payload \n " ,
2018-06-23 14:52:24 -03:00
__func__ ) ;
return rc ;
}
drqst . rq_iov + + ;
drqst . rq_nvec - - ;
}
2012-09-18 16:20:30 -07:00
2018-06-23 14:52:24 -03:00
rc = __cifs_calc_signature ( & drqst , server , sigptr , shash ) ;
2015-11-12 22:46:49 -05:00
if ( ! rc )
2016-10-24 15:33:04 -07:00
memcpy ( shdr - > Signature , sigptr , SMB2_SIGNATURE_SIZE ) ;
2012-09-18 16:20:30 -07:00
return rc ;
}
2015-12-18 13:05:30 -06:00
static int generate_key ( struct cifs_ses * ses , struct kvec label ,
struct kvec context , __u8 * key , unsigned int key_size )
2013-06-26 23:45:05 -05:00
{
unsigned char zero = 0x0 ;
__u8 i [ 4 ] = { 0 , 0 , 0 , 1 } ;
__u8 L [ 4 ] = { 0 , 0 , 0 , 128 } ;
int rc = 0 ;
unsigned char prfhash [ SMB2_HMACSHA256_SIZE ] ;
unsigned char * hashptr = prfhash ;
2019-08-28 17:15:35 +10:00
struct TCP_Server_Info * server = ses - > server ;
2013-06-26 23:45:05 -05:00
memset ( prfhash , 0x0 , SMB2_HMACSHA256_SIZE ) ;
2015-12-18 13:05:30 -06:00
memset ( key , 0x0 , key_size ) ;
2013-06-26 23:45:05 -05:00
2019-08-28 17:15:35 +10:00
rc = smb3_crypto_shash_allocate ( server ) ;
2013-07-04 10:35:21 -05:00
if ( rc ) {
2019-08-28 17:15:35 +10:00
cifs_server_dbg ( VFS , " %s: crypto alloc failed \n " , __func__ ) ;
2013-07-04 10:35:21 -05:00
goto smb3signkey_ret ;
}
2019-08-28 17:15:35 +10:00
rc = crypto_shash_setkey ( server - > secmech . hmacsha256 ,
2013-08-29 08:35:11 -05:00
ses - > auth_key . response , SMB2_NTLMV2_SESSKEY_SIZE ) ;
2013-06-26 23:45:05 -05:00
if ( rc ) {
2019-08-28 17:15:35 +10:00
cifs_server_dbg ( VFS , " %s: Could not set with session key \n " , __func__ ) ;
2013-06-26 23:45:05 -05:00
goto smb3signkey_ret ;
}
2019-08-28 17:15:35 +10:00
rc = crypto_shash_init ( & server - > secmech . sdeschmacsha256 - > shash ) ;
2013-06-26 23:45:05 -05:00
if ( rc ) {
2019-08-28 17:15:35 +10:00
cifs_server_dbg ( VFS , " %s: Could not init sign hmac \n " , __func__ ) ;
2013-06-26 23:45:05 -05:00
goto smb3signkey_ret ;
}
2019-08-28 17:15:35 +10:00
rc = crypto_shash_update ( & server - > secmech . sdeschmacsha256 - > shash ,
2013-06-26 23:45:05 -05:00
i , 4 ) ;
if ( rc ) {
2019-08-28 17:15:35 +10:00
cifs_server_dbg ( VFS , " %s: Could not update with n \n " , __func__ ) ;
2013-06-26 23:45:05 -05:00
goto smb3signkey_ret ;
}
2019-08-28 17:15:35 +10:00
rc = crypto_shash_update ( & server - > secmech . sdeschmacsha256 - > shash ,
2015-12-18 13:05:30 -06:00
label . iov_base , label . iov_len ) ;
2013-06-26 23:45:05 -05:00
if ( rc ) {
2019-08-28 17:15:35 +10:00
cifs_server_dbg ( VFS , " %s: Could not update with label \n " , __func__ ) ;
2013-06-26 23:45:05 -05:00
goto smb3signkey_ret ;
}
2019-08-28 17:15:35 +10:00
rc = crypto_shash_update ( & server - > secmech . sdeschmacsha256 - > shash ,
2013-06-26 23:45:05 -05:00
& zero , 1 ) ;
if ( rc ) {
2019-08-28 17:15:35 +10:00
cifs_server_dbg ( VFS , " %s: Could not update with zero \n " , __func__ ) ;
2013-06-26 23:45:05 -05:00
goto smb3signkey_ret ;
}
2019-08-28 17:15:35 +10:00
rc = crypto_shash_update ( & server - > secmech . sdeschmacsha256 - > shash ,
2015-12-18 13:05:30 -06:00
context . iov_base , context . iov_len ) ;
2013-06-26 23:45:05 -05:00
if ( rc ) {
2019-08-28 17:15:35 +10:00
cifs_server_dbg ( VFS , " %s: Could not update with context \n " , __func__ ) ;
2013-06-26 23:45:05 -05:00
goto smb3signkey_ret ;
}
2019-08-28 17:15:35 +10:00
rc = crypto_shash_update ( & server - > secmech . sdeschmacsha256 - > shash ,
2013-06-26 23:45:05 -05:00
L , 4 ) ;
if ( rc ) {
2019-08-28 17:15:35 +10:00
cifs_server_dbg ( VFS , " %s: Could not update with L \n " , __func__ ) ;
2013-06-26 23:45:05 -05:00
goto smb3signkey_ret ;
}
2019-08-28 17:15:35 +10:00
rc = crypto_shash_final ( & server - > secmech . sdeschmacsha256 - > shash ,
2013-06-26 23:45:05 -05:00
hashptr ) ;
if ( rc ) {
2019-08-28 17:15:35 +10:00
cifs_server_dbg ( VFS , " %s: Could not generate sha256 hash \n " , __func__ ) ;
2013-06-26 23:45:05 -05:00
goto smb3signkey_ret ;
}
2015-12-18 13:05:30 -06:00
memcpy ( key , hashptr , key_size ) ;
2013-06-26 23:45:05 -05:00
smb3signkey_ret :
2013-08-29 08:35:11 -05:00
return rc ;
2013-06-26 23:45:05 -05:00
}
2015-12-18 13:05:30 -06:00
struct derivation {
struct kvec label ;
struct kvec context ;
} ;
struct derivation_triplet {
struct derivation signing ;
struct derivation encryption ;
struct derivation decryption ;
} ;
static int
generate_smb3signingkey ( struct cifs_ses * ses ,
const struct derivation_triplet * ptriplet )
{
int rc ;
2019-09-20 06:31:10 +02:00
/*
* All channels use the same encryption / decryption keys but
* they have their own signing key .
*
* When we generate the keys , check if it is for a new channel
* ( binding ) in which case we only need to generate a signing
* key and store it in the channel as to not overwrite the
* master connection signing key stored in the session
*/
2015-12-18 13:05:30 -06:00
2019-09-20 06:31:10 +02:00
if ( ses - > binding ) {
rc = generate_key ( ses , ptriplet - > signing . label ,
ptriplet - > signing . context ,
cifs_ses_binding_channel ( ses ) - > signkey ,
SMB3_SIGN_KEY_SIZE ) ;
if ( rc )
return rc ;
} else {
rc = generate_key ( ses , ptriplet - > signing . label ,
ptriplet - > signing . context ,
ses - > smb3signingkey ,
SMB3_SIGN_KEY_SIZE ) ;
if ( rc )
return rc ;
2019-11-22 12:30:57 -03:00
memcpy ( ses - > chans [ 0 ] . signkey , ses - > smb3signingkey ,
SMB3_SIGN_KEY_SIZE ) ;
2019-09-20 06:31:10 +02:00
rc = generate_key ( ses , ptriplet - > encryption . label ,
ptriplet - > encryption . context ,
ses - > smb3encryptionkey ,
SMB3_SIGN_KEY_SIZE ) ;
rc = generate_key ( ses , ptriplet - > decryption . label ,
ptriplet - > decryption . context ,
ses - > smb3decryptionkey ,
SMB3_SIGN_KEY_SIZE ) ;
if ( rc )
return rc ;
}
2017-05-24 16:13:25 +02:00
if ( rc )
return rc ;
# ifdef CONFIG_CIFS_DEBUG_DUMP_KEYS
cifs_dbg ( VFS , " %s: dumping generated AES session keys \n " , __func__ ) ;
/*
* The session id is opaque in terms of endianness , so we can ' t
* print it as a long long . we dump it as we got it on the wire
*/
cifs_dbg ( VFS , " Session Id %*ph \n " , ( int ) sizeof ( ses - > Suid ) ,
& ses - > Suid ) ;
cifs_dbg ( VFS , " Session Key %*ph \n " ,
SMB2_NTLMV2_SESSKEY_SIZE , ses - > auth_key . response ) ;
cifs_dbg ( VFS , " Signing Key %*ph \n " ,
SMB3_SIGN_KEY_SIZE , ses - > smb3signingkey ) ;
cifs_dbg ( VFS , " ServerIn Key %*ph \n " ,
SMB3_SIGN_KEY_SIZE , ses - > smb3encryptionkey ) ;
cifs_dbg ( VFS , " ServerOut Key %*ph \n " ,
SMB3_SIGN_KEY_SIZE , ses - > smb3decryptionkey ) ;
# endif
return rc ;
2015-12-18 13:05:30 -06:00
}
int
generate_smb30signingkey ( struct cifs_ses * ses )
{
struct derivation_triplet triplet ;
struct derivation * d ;
d = & triplet . signing ;
d - > label . iov_base = " SMB2AESCMAC " ;
d - > label . iov_len = 12 ;
d - > context . iov_base = " SmbSign " ;
d - > context . iov_len = 8 ;
d = & triplet . encryption ;
d - > label . iov_base = " SMB2AESCCM " ;
d - > label . iov_len = 11 ;
d - > context . iov_base = " ServerIn " ;
d - > context . iov_len = 10 ;
d = & triplet . decryption ;
d - > label . iov_base = " SMB2AESCCM " ;
d - > label . iov_len = 11 ;
d - > context . iov_base = " ServerOut " ;
d - > context . iov_len = 10 ;
return generate_smb3signingkey ( ses , & triplet ) ;
}
int
generate_smb311signingkey ( struct cifs_ses * ses )
{
struct derivation_triplet triplet ;
struct derivation * d ;
d = & triplet . signing ;
2017-09-25 20:11:58 -05:00
d - > label . iov_base = " SMBSigningKey " ;
d - > label . iov_len = 14 ;
d - > context . iov_base = ses - > preauth_sha_hash ;
d - > context . iov_len = 64 ;
2015-12-18 13:05:30 -06:00
d = & triplet . encryption ;
2017-09-25 20:11:58 -05:00
d - > label . iov_base = " SMBC2SCipherKey " ;
d - > label . iov_len = 16 ;
d - > context . iov_base = ses - > preauth_sha_hash ;
d - > context . iov_len = 64 ;
2015-12-18 13:05:30 -06:00
d = & triplet . decryption ;
2017-09-25 20:11:58 -05:00
d - > label . iov_base = " SMBS2CCipherKey " ;
d - > label . iov_len = 16 ;
d - > context . iov_base = ses - > preauth_sha_hash ;
d - > context . iov_len = 64 ;
2015-12-18 13:05:30 -06:00
return generate_smb3signingkey ( ses , & triplet ) ;
}
2012-12-08 22:08:06 -06:00
int
smb3_calc_signature ( struct smb_rqst * rqst , struct TCP_Server_Info * server )
{
2018-06-23 14:52:23 -03:00
int rc ;
2013-06-26 23:45:05 -05:00
unsigned char smb3_signature [ SMB2_CMACAES_SIZE ] ;
unsigned char * sigptr = smb3_signature ;
struct kvec * iov = rqst - > rq_iov ;
2018-06-12 08:00:58 +10:00
struct smb2_sync_hdr * shdr = ( struct smb2_sync_hdr * ) iov [ 0 ] . iov_base ;
2018-06-23 14:52:23 -03:00
struct shash_desc * shash = & server - > secmech . sdesccmacaes - > shash ;
struct smb_rqst drqst ;
2019-09-20 06:31:10 +02:00
u8 key [ SMB3_SIGN_KEY_SIZE ] ;
2013-08-29 08:35:11 -05:00
2019-09-20 06:31:10 +02:00
rc = smb2_get_sign_key ( shdr - > SessionId , server , key ) ;
if ( rc )
2013-08-29 08:35:11 -05:00
return 0 ;
2013-06-26 23:45:05 -05:00
memset ( smb3_signature , 0x0 , SMB2_CMACAES_SIZE ) ;
2016-10-24 15:33:04 -07:00
memset ( shdr - > Signature , 0x0 , SMB2_SIGNATURE_SIZE ) ;
2013-06-26 23:45:05 -05:00
rc = crypto_shash_setkey ( server - > secmech . cmacaes ,
2019-09-20 06:31:10 +02:00
key , SMB2_CMACAES_SIZE ) ;
2013-06-26 23:45:05 -05:00
if ( rc ) {
2019-08-28 17:15:35 +10:00
cifs_server_dbg ( VFS , " %s: Could not set key for cmac aes \n " , __func__ ) ;
2013-06-26 23:45:05 -05:00
return rc ;
}
2013-07-04 10:35:21 -05:00
/*
* we already allocate sdesccmacaes when we init smb3 signing key ,
* so unlike smb2 case we do not have to check here if secmech are
* initialized
*/
2018-06-23 14:52:23 -03:00
rc = crypto_shash_init ( shash ) ;
2013-06-26 23:45:05 -05:00
if ( rc ) {
2019-08-28 17:15:35 +10:00
cifs_server_dbg ( VFS , " %s: Could not init cmac aes \n " , __func__ ) ;
2013-06-26 23:45:05 -05:00
return rc ;
}
2018-02-16 19:19:27 +01:00
2018-06-23 14:52:23 -03:00
/*
* For SMB2 + , __cifs_calc_signature ( ) expects to sign only the actual
* data , that is , iov [ 0 ] should not contain a rfc1002 length .
*
* Sign the rfc1002 length prior to passing the data ( iov [ 1 - N ] ) down to
* __cifs_calc_signature ( ) .
*/
drqst = * rqst ;
if ( drqst . rq_nvec > = 2 & & iov [ 0 ] . iov_len = = 4 ) {
rc = crypto_shash_update ( shash , iov [ 0 ] . iov_base ,
iov [ 0 ] . iov_len ) ;
if ( rc ) {
2019-08-28 17:15:35 +10:00
cifs_server_dbg ( VFS , " %s: Could not update with payload \n " ,
2018-06-23 14:52:23 -03:00
__func__ ) ;
return rc ;
}
drqst . rq_iov + + ;
drqst . rq_nvec - - ;
}
2013-06-26 23:45:05 -05:00
2018-06-23 14:52:23 -03:00
rc = __cifs_calc_signature ( & drqst , server , sigptr , shash ) ;
2015-11-12 22:46:49 -05:00
if ( ! rc )
2016-10-24 15:33:04 -07:00
memcpy ( shdr - > Signature , sigptr , SMB2_SIGNATURE_SIZE ) ;
2013-06-26 23:45:05 -05:00
return rc ;
2012-12-08 22:08:06 -06:00
}
2012-09-18 16:20:30 -07:00
/* must be called with server->srv_mutex held */
static int
2012-09-18 16:20:34 -07:00
smb2_sign_rqst ( struct smb_rqst * rqst , struct TCP_Server_Info * server )
2012-09-18 16:20:30 -07:00
{
int rc = 0 ;
2019-09-20 06:31:10 +02:00
struct smb2_sync_hdr * shdr ;
struct smb2_sess_setup_req * ssr ;
bool is_binding ;
bool is_signed ;
2012-09-18 16:20:30 -07:00
2019-09-20 06:31:10 +02:00
shdr = ( struct smb2_sync_hdr * ) rqst - > rq_iov [ 0 ] . iov_base ;
ssr = ( struct smb2_sess_setup_req * ) shdr ;
2012-09-18 16:20:30 -07:00
2019-09-20 06:31:10 +02:00
is_binding = shdr - > Command = = SMB2_SESSION_SETUP & &
( ssr - > Flags & SMB2_SESSION_REQ_FLAG_BINDING ) ;
is_signed = shdr - > Flags & SMB2_FLAGS_SIGNED ;
if ( ! is_signed )
return 0 ;
if ( server - > tcpStatus = = CifsNeedNegotiate )
return 0 ;
if ( ! is_binding & & ! server - > session_estab ) {
2016-10-24 15:33:04 -07:00
strncpy ( shdr - > Signature , " BSRSPYL " , 8 ) ;
2019-09-20 06:31:10 +02:00
return 0 ;
2012-09-18 16:20:30 -07:00
}
2012-12-08 22:08:06 -06:00
rc = server - > ops - > calc_signature ( rqst , server ) ;
2012-09-18 16:20:30 -07:00
return rc ;
}
int
2012-09-18 16:20:34 -07:00
smb2_verify_signature ( struct smb_rqst * rqst , struct TCP_Server_Info * server )
2012-09-18 16:20:30 -07:00
{
unsigned int rc ;
char server_response_sig [ 16 ] ;
2016-11-23 15:14:57 -08:00
struct smb2_sync_hdr * shdr =
2018-06-01 10:53:02 +10:00
( struct smb2_sync_hdr * ) rqst - > rq_iov [ 0 ] . iov_base ;
2012-09-18 16:20:30 -07:00
2016-10-24 15:33:04 -07:00
if ( ( shdr - > Command = = SMB2_NEGOTIATE ) | |
( shdr - > Command = = SMB2_SESSION_SETUP ) | |
( shdr - > Command = = SMB2_OPLOCK_BREAK ) | |
2019-09-03 21:18:49 -05:00
server - > ignore_signature | |
2012-09-18 16:20:30 -07:00
( ! server - > session_estab ) )
return 0 ;
/*
* BB what if signatures are supposed to be on for session but
* server does not send one ? BB
*/
/* Do not need to verify session setups with signature "BSRSPYL " */
2016-10-24 15:33:04 -07:00
if ( memcmp ( shdr - > Signature , " BSRSPYL " , 8 ) = = 0 )
2013-05-04 22:12:25 -05:00
cifs_dbg ( FYI , " dummy signature received for smb command 0x%x \n " ,
2016-10-24 15:33:04 -07:00
shdr - > Command ) ;
2012-09-18 16:20:30 -07:00
/*
* Save off the origiginal signature so we can modify the smb and check
* our calculated signature against what the server sent .
*/
2016-10-24 15:33:04 -07:00
memcpy ( server_response_sig , shdr - > Signature , SMB2_SIGNATURE_SIZE ) ;
2012-09-18 16:20:30 -07:00
2016-10-24 15:33:04 -07:00
memset ( shdr - > Signature , 0 , SMB2_SIGNATURE_SIZE ) ;
2012-09-18 16:20:30 -07:00
mutex_lock ( & server - > srv_mutex ) ;
2012-12-08 22:08:06 -06:00
rc = server - > ops - > calc_signature ( rqst , server ) ;
2012-09-18 16:20:30 -07:00
mutex_unlock ( & server - > srv_mutex ) ;
if ( rc )
return rc ;
2016-10-24 15:33:04 -07:00
if ( memcmp ( server_response_sig , shdr - > Signature , SMB2_SIGNATURE_SIZE ) )
2012-09-18 16:20:30 -07:00
return - EACCES ;
else
return 0 ;
}
2011-12-26 22:53:34 +04:00
/*
* Set message id for the request . Should be called after wait_for_free_request
* and when srv_mutex is held .
*/
static inline void
2016-10-24 15:33:04 -07:00
smb2_seq_num_into_buf ( struct TCP_Server_Info * server ,
struct smb2_sync_hdr * shdr )
2011-12-26 22:53:34 +04:00
{
2016-10-24 15:33:04 -07:00
unsigned int i , num = le16_to_cpu ( shdr - > CreditCharge ) ;
2014-06-05 19:03:27 +04:00
2016-10-24 15:33:04 -07:00
shdr - > MessageId = get_next_mid64 ( server ) ;
2014-06-05 19:03:27 +04:00
/* skip message numbers according to CreditCharge field */
for ( i = 1 ; i < num ; i + + )
get_next_mid ( server ) ;
2011-12-26 22:53:34 +04:00
}
static struct mid_q_entry *
2016-10-24 15:33:04 -07:00
smb2_mid_entry_alloc ( const struct smb2_sync_hdr * shdr ,
2011-12-26 22:53:34 +04:00
struct TCP_Server_Info * server )
{
struct mid_q_entry * temp ;
2019-03-04 14:02:50 -08:00
unsigned int credits = le16_to_cpu ( shdr - > CreditCharge ) ;
2011-12-26 22:53:34 +04:00
if ( server = = NULL ) {
2013-05-04 22:12:25 -05:00
cifs_dbg ( VFS , " Null TCP session in smb2_mid_entry_alloc \n " ) ;
2011-12-26 22:53:34 +04:00
return NULL ;
}
temp = mempool_alloc ( cifs_mid_poolp , GFP_NOFS ) ;
2017-04-10 12:08:53 +10:00
memset ( temp , 0 , sizeof ( struct mid_q_entry ) ) ;
2018-06-25 14:05:25 +02:00
kref_init ( & temp - > refcount ) ;
2017-04-10 12:08:53 +10:00
temp - > mid = le64_to_cpu ( shdr - > MessageId ) ;
2019-03-04 14:02:50 -08:00
temp - > credits = credits > 0 ? credits : 1 ;
2017-04-10 12:08:53 +10:00
temp - > pid = current - > pid ;
temp - > command = shdr - > Command ; /* Always LE */
temp - > when_alloc = jiffies ;
temp - > server = server ;
/*
* The default is for the mid to be synchronous , so the
* default callback just wakes up the current task .
*/
temp - > callback = cifs_wake_up_task ;
temp - > callback_data = current ;
2011-12-26 22:53:34 +04:00
atomic_inc ( & midCount ) ;
temp - > mid_state = MID_REQUEST_ALLOCATED ;
2019-02-26 21:26:20 -06:00
trace_smb3_cmd_enter ( shdr - > TreeId , shdr - > SessionId ,
le16_to_cpu ( shdr - > Command ) , temp - > mid ) ;
2011-12-26 22:53:34 +04:00
return temp ;
}
static int
2019-09-20 06:08:34 +02:00
smb2_get_mid_entry ( struct cifs_ses * ses , struct TCP_Server_Info * server ,
struct smb2_sync_hdr * shdr , struct mid_q_entry * * mid )
2011-12-26 22:53:34 +04:00
{
2019-09-20 06:08:34 +02:00
if ( server - > tcpStatus = = CifsExiting )
2011-12-26 22:53:34 +04:00
return - ENOENT ;
2019-09-20 06:08:34 +02:00
if ( server - > tcpStatus = = CifsNeedReconnect ) {
2013-05-04 22:12:25 -05:00
cifs_dbg ( FYI , " tcp session dead - return to caller to retry \n " ) ;
2011-12-26 22:53:34 +04:00
return - EAGAIN ;
}
2019-09-20 06:08:34 +02:00
if ( server - > tcpStatus = = CifsNeedNegotiate & &
2019-03-05 15:51:55 -08:00
shdr - > Command ! = SMB2_NEGOTIATE )
return - EAGAIN ;
2013-10-12 10:06:03 -05:00
if ( ses - > status = = CifsNew ) {
2016-10-24 15:33:04 -07:00
if ( ( shdr - > Command ! = SMB2_SESSION_SETUP ) & &
( shdr - > Command ! = SMB2_NEGOTIATE ) )
2011-12-26 22:53:34 +04:00
return - EAGAIN ;
/* else ok - we are setting up session */
}
2013-10-12 10:06:03 -05:00
if ( ses - > status = = CifsExiting ) {
2016-10-24 15:33:04 -07:00
if ( shdr - > Command ! = SMB2_LOGOFF )
2013-10-12 10:06:03 -05:00
return - EAGAIN ;
/* else ok - we are shutting down the session */
}
2019-09-20 06:08:34 +02:00
* mid = smb2_mid_entry_alloc ( shdr , server ) ;
2011-12-26 22:53:34 +04:00
if ( * mid = = NULL )
return - ENOMEM ;
spin_lock ( & GlobalMid_Lock ) ;
2019-09-20 06:08:34 +02:00
list_add_tail ( & ( * mid ) - > qhead , & server - > pending_mid_q ) ;
2011-12-26 22:53:34 +04:00
spin_unlock ( & GlobalMid_Lock ) ;
2019-02-26 21:26:20 -06:00
2011-12-26 22:53:34 +04:00
return 0 ;
}
int
smb2_check_receive ( struct mid_q_entry * mid , struct TCP_Server_Info * server ,
bool log_error )
{
2018-04-09 18:06:28 +10:00
unsigned int len = mid - > resp_buf_size ;
2018-06-01 10:53:02 +10:00
struct kvec iov [ 1 ] ;
2016-11-23 15:14:57 -08:00
struct smb_rqst rqst = { . rq_iov = iov ,
2018-06-01 10:53:02 +10:00
. rq_nvec = 1 } ;
2012-09-18 16:20:34 -07:00
2016-11-23 15:14:57 -08:00
iov [ 0 ] . iov_base = ( char * ) mid - > resp_buf ;
2018-06-01 10:53:02 +10:00
iov [ 0 ] . iov_len = len ;
2011-12-26 22:53:34 +04:00
dump_smb ( mid - > resp_buf , min_t ( u32 , 80 , len ) ) ;
/* convert the length into a more usable form */
2016-11-17 15:24:46 -08:00
if ( len > 24 & & server - > sign & & ! mid - > decrypted ) {
2012-09-18 16:20:30 -07:00
int rc ;
2012-09-18 16:20:34 -07:00
rc = smb2_verify_signature ( & rqst , server ) ;
2012-09-18 16:20:30 -07:00
if ( rc )
2019-08-28 17:15:35 +10:00
cifs_server_dbg ( VFS , " SMB signature verification returned error = %d \n " ,
2013-05-04 22:12:25 -05:00
rc ) ;
2012-09-18 16:20:30 -07:00
}
2011-12-26 22:53:34 +04:00
return map_smb2_to_linux_error ( mid - > resp_buf , log_error ) ;
}
2012-09-18 16:20:35 -07:00
struct mid_q_entry *
2019-09-20 06:08:34 +02:00
smb2_setup_request ( struct cifs_ses * ses , struct TCP_Server_Info * server ,
struct smb_rqst * rqst )
2011-12-26 22:53:34 +04:00
{
int rc ;
2016-11-23 15:14:57 -08:00
struct smb2_sync_hdr * shdr =
2018-06-12 08:00:58 +10:00
( struct smb2_sync_hdr * ) rqst - > rq_iov [ 0 ] . iov_base ;
2011-12-26 22:53:34 +04:00
struct mid_q_entry * mid ;
2019-09-20 06:08:34 +02:00
smb2_seq_num_into_buf ( server , shdr ) ;
2011-12-26 22:53:34 +04:00
2019-09-20 06:08:34 +02:00
rc = smb2_get_mid_entry ( ses , server , shdr , & mid ) ;
2019-03-04 14:02:50 -08:00
if ( rc ) {
2019-09-20 06:08:34 +02:00
revert_current_mid_from_hdr ( server , shdr ) ;
2012-09-18 16:20:35 -07:00
return ERR_PTR ( rc ) ;
2019-03-04 14:02:50 -08:00
}
2019-09-20 06:08:34 +02:00
rc = smb2_sign_rqst ( rqst , server ) ;
2012-09-18 16:20:35 -07:00
if ( rc ) {
2019-09-20 06:08:34 +02:00
revert_current_mid_from_hdr ( server , shdr ) ;
2012-09-18 16:20:30 -07:00
cifs_delete_mid ( mid ) ;
2012-09-18 16:20:35 -07:00
return ERR_PTR ( rc ) ;
}
2019-03-04 14:02:50 -08:00
2012-09-18 16:20:35 -07:00
return mid ;
2011-12-26 22:53:34 +04:00
}
2012-09-18 16:20:35 -07:00
struct mid_q_entry *
smb2_setup_async_request ( struct TCP_Server_Info * server , struct smb_rqst * rqst )
2012-07-11 14:45:28 +04:00
{
2012-09-18 16:20:35 -07:00
int rc ;
2016-11-23 15:14:57 -08:00
struct smb2_sync_hdr * shdr =
2018-06-12 08:00:58 +10:00
( struct smb2_sync_hdr * ) rqst - > rq_iov [ 0 ] . iov_base ;
2012-07-11 14:45:28 +04:00
struct mid_q_entry * mid ;
2019-03-05 15:51:55 -08:00
if ( server - > tcpStatus = = CifsNeedNegotiate & &
shdr - > Command ! = SMB2_NEGOTIATE )
return ERR_PTR ( - EAGAIN ) ;
2016-10-24 15:33:04 -07:00
smb2_seq_num_into_buf ( server , shdr ) ;
2012-07-11 14:45:28 +04:00
2016-10-24 15:33:04 -07:00
mid = smb2_mid_entry_alloc ( shdr , server ) ;
2019-03-04 14:02:50 -08:00
if ( mid = = NULL ) {
revert_current_mid_from_hdr ( server , shdr ) ;
2012-09-18 16:20:35 -07:00
return ERR_PTR ( - ENOMEM ) ;
2019-03-04 14:02:50 -08:00
}
2012-07-11 14:45:28 +04:00
2012-09-18 16:20:35 -07:00
rc = smb2_sign_rqst ( rqst , server ) ;
2012-07-11 14:45:28 +04:00
if ( rc ) {
2019-03-04 14:02:50 -08:00
revert_current_mid_from_hdr ( server , shdr ) ;
2012-07-11 14:45:28 +04:00
DeleteMidQEntry ( mid ) ;
2012-09-18 16:20:35 -07:00
return ERR_PTR ( rc ) ;
2012-09-18 16:20:30 -07:00
}
2012-09-18 16:20:35 -07:00
return mid ;
2012-07-11 14:45:28 +04:00
}
2016-11-03 16:47:37 -07:00
int
smb3_crypto_aead_allocate ( struct TCP_Server_Info * server )
{
struct crypto_aead * tfm ;
if ( ! server - > secmech . ccmaesencrypt ) {
2019-06-07 15:16:10 -05:00
if ( server - > cipher_type = = SMB2_ENCRYPTION_AES128_GCM )
tfm = crypto_alloc_aead ( " gcm(aes) " , 0 , 0 ) ;
else
tfm = crypto_alloc_aead ( " ccm(aes) " , 0 , 0 ) ;
2016-11-03 16:47:37 -07:00
if ( IS_ERR ( tfm ) ) {
2019-08-28 17:15:35 +10:00
cifs_server_dbg ( VFS , " %s: Failed to alloc encrypt aead \n " ,
2016-11-03 16:47:37 -07:00
__func__ ) ;
return PTR_ERR ( tfm ) ;
}
server - > secmech . ccmaesencrypt = tfm ;
}
if ( ! server - > secmech . ccmaesdecrypt ) {
2019-06-07 15:16:10 -05:00
if ( server - > cipher_type = = SMB2_ENCRYPTION_AES128_GCM )
tfm = crypto_alloc_aead ( " gcm(aes) " , 0 , 0 ) ;
else
tfm = crypto_alloc_aead ( " ccm(aes) " , 0 , 0 ) ;
2016-11-03 16:47:37 -07:00
if ( IS_ERR ( tfm ) ) {
crypto_free_aead ( server - > secmech . ccmaesencrypt ) ;
server - > secmech . ccmaesencrypt = NULL ;
2019-08-28 17:15:35 +10:00
cifs_server_dbg ( VFS , " %s: Failed to alloc decrypt aead \n " ,
2016-11-03 16:47:37 -07:00
__func__ ) ;
return PTR_ERR ( tfm ) ;
}
server - > secmech . ccmaesdecrypt = tfm ;
}
return 0 ;
}