CVE-2016-2114: s3:smbd: use the correct default values for "smb signing"
[samba.git] / source3 / smbd / smb2_negprot.c
index 11ec2a5eff4dda7ac65159bface61ffbc22e887a..9c03b2ca8b86158db1814aff4c7dc828dd253c8b 100644 (file)
 #include "smbd/smbd.h"
 #include "smbd/globals.h"
 #include "../libcli/smb/smb_common.h"
+#include "../libcli/smb/smb2_negotiate_context.h"
 #include "../lib/tsocket/tsocket.h"
 #include "../librpc/ndr/libndr.h"
+#include "../libcli/smb/smb_signing.h"
+
+extern fstring remote_proto;
 
 /*
  * this is the entry point if SMB2 is selected via
  */
 static void reply_smb20xx(struct smb_request *req, uint16_t dialect)
 {
-       uint8_t *smb2_inbuf;
+       uint8_t *smb2_inpdu;
        uint8_t *smb2_hdr;
        uint8_t *smb2_body;
        uint8_t *smb2_dyn;
-       size_t len = 4 + SMB2_HDR_BODY + 0x24 + 2;
+       size_t len = SMB2_HDR_BODY + 0x24 + 2;
 
-       smb2_inbuf = talloc_zero_array(talloc_tos(), uint8_t, len);
-       if (smb2_inbuf == NULL) {
+       smb2_inpdu = talloc_zero_array(talloc_tos(), uint8_t, len);
+       if (smb2_inpdu == NULL) {
                DEBUG(0, ("Could not push spnego blob\n"));
                reply_nterror(req, NT_STATUS_NO_MEMORY);
                return;
        }
-       smb2_hdr = smb2_inbuf + 4;
+       smb2_hdr = smb2_inpdu;
        smb2_body = smb2_hdr + SMB2_HDR_BODY;
        smb2_dyn = smb2_body + 0x24;
 
@@ -57,7 +61,7 @@ static void reply_smb20xx(struct smb_request *req, uint16_t dialect)
 
        req->outbuf = NULL;
 
-       smbd_smb2_first_negprot(req->sconn, smb2_inbuf, len);
+       smbd_smb2_process_negprot(req->xconn, 0, smb2_inpdu, len);
        return;
 }
 
@@ -76,12 +80,54 @@ void reply_smb2002(struct smb_request *req, uint16_t choice)
  */
 void reply_smb20ff(struct smb_request *req, uint16_t choice)
 {
-       req->sconn->smb2.negprot_2ff = true;
+       struct smbXsrv_connection *xconn = req->xconn;
+       xconn->smb2.allow_2ff = true;
        reply_smb20xx(req, SMB2_DIALECT_REVISION_2FF);
 }
 
+enum protocol_types smbd_smb2_protocol_dialect_match(const uint8_t *indyn,
+                               const int dialect_count,
+                               uint16_t *dialect)
+{
+       struct {
+               enum protocol_types proto;
+               uint16_t dialect;
+       } pd[] = {
+               { PROTOCOL_SMB3_11, SMB3_DIALECT_REVISION_311 },
+               { PROTOCOL_SMB3_10, SMB3_DIALECT_REVISION_310 },
+               { PROTOCOL_SMB3_02, SMB3_DIALECT_REVISION_302 },
+               { PROTOCOL_SMB3_00, SMB3_DIALECT_REVISION_300 },
+               { PROTOCOL_SMB2_24, SMB2_DIALECT_REVISION_224 },
+               { PROTOCOL_SMB2_22, SMB2_DIALECT_REVISION_222 },
+               { PROTOCOL_SMB2_10, SMB2_DIALECT_REVISION_210 },
+               { PROTOCOL_SMB2_02, SMB2_DIALECT_REVISION_202 },
+       };
+       size_t i;
+
+       for (i = 0; i < ARRAY_SIZE(pd); i ++) {
+               size_t c = 0;
+
+               if (lp_server_max_protocol() < pd[i].proto) {
+                       continue;
+               }
+               if (lp_server_min_protocol() > pd[i].proto) {
+                       continue;
+               }
+
+               for (c = 0; c < dialect_count; c++) {
+                       *dialect = SVAL(indyn, c*2);
+                       if (*dialect == pd[i].dialect) {
+                               return pd[i].proto;
+                       }
+               }
+       }
+
+       return PROTOCOL_NONE;
+}
+
 NTSTATUS smbd_smb2_request_process_negprot(struct smbd_smb2_request *req)
 {
+       struct smbXsrv_connection *xconn = req->xconn;
        NTSTATUS status;
        const uint8_t *inbody;
        const uint8_t *indyn = NULL;
@@ -98,6 +144,13 @@ NTSTATUS smbd_smb2_request_process_negprot(struct smbd_smb2_request *req)
        uint32_t in_capabilities;
        DATA_BLOB in_guid_blob;
        struct GUID in_guid;
+       struct smb2_negotiate_contexts in_c = { .num_contexts = 0, };
+       struct smb2_negotiate_context *in_preauth = NULL;
+       struct smb2_negotiate_context *in_cipher = NULL;
+       struct smb2_negotiate_contexts out_c = { .num_contexts = 0, };
+       DATA_BLOB out_negotiate_context_blob = data_blob_null;
+       uint32_t out_negotiate_context_offset = 0;
+       uint16_t out_negotiate_context_count = 0;
        uint16_t dialect = 0;
        uint32_t capabilities;
        DATA_BLOB out_guid_blob;
@@ -108,6 +161,7 @@ NTSTATUS smbd_smb2_request_process_negprot(struct smbd_smb2_request *req)
        uint32_t max_read = lp_smb2_max_read();
        uint32_t max_write = lp_smb2_max_write();
        NTTIME now = timeval_to_nttime(&req->request_time);
+       bool signing_required = true;
 
        status = smbd_smb2_request_verify_sizes(req, 0x24);
        if (!NT_STATUS_IS_OK(status)) {
@@ -136,106 +190,103 @@ NTSTATUS smbd_smb2_request_process_negprot(struct smbd_smb2_request *req)
        }
        indyn = SMBD_SMB2_IN_DYN_PTR(req);
 
+       protocol = smbd_smb2_protocol_dialect_match(indyn,
+                                       dialect_count,
+                                       &dialect);
+
        for (c=0; protocol == PROTOCOL_NONE && c < dialect_count; c++) {
-               if (lp_srv_maxprotocol() < PROTOCOL_SMB3_00) {
-                       break;
-               }
-               if (lp_srv_minprotocol() > PROTOCOL_SMB3_00) {
+               if (lp_server_max_protocol() < PROTOCOL_SMB2_10) {
                        break;
                }
 
                dialect = SVAL(indyn, c*2);
-               if (dialect == SMB3_DIALECT_REVISION_300) {
-                       protocol = PROTOCOL_SMB3_00;
-                       break;
+               if (dialect == SMB2_DIALECT_REVISION_2FF) {
+                       if (xconn->smb2.allow_2ff) {
+                               xconn->smb2.allow_2ff = false;
+                               protocol = PROTOCOL_SMB2_10;
+                               break;
+                       }
                }
        }
 
-       for (c=0; protocol == PROTOCOL_NONE && c < dialect_count; c++) {
-               if (lp_srv_maxprotocol() < PROTOCOL_SMB2_24) {
-                       break;
-               }
-               if (lp_srv_minprotocol() > PROTOCOL_SMB2_24) {
-                       break;
-               }
-
-               dialect = SVAL(indyn, c*2);
-               if (dialect == SMB2_DIALECT_REVISION_224) {
-                       protocol = PROTOCOL_SMB2_24;
-                       break;
-               }
+       if (protocol == PROTOCOL_NONE) {
+               return smbd_smb2_request_error(req, NT_STATUS_NOT_SUPPORTED);
        }
 
-       for (c=0; protocol == PROTOCOL_NONE && c < dialect_count; c++) {
-               if (lp_srv_maxprotocol() < PROTOCOL_SMB2_22) {
-                       break;
-               }
-               if (lp_srv_minprotocol() > PROTOCOL_SMB2_22) {
-                       break;
-               }
+       if (protocol >= PROTOCOL_SMB3_10) {
+               uint32_t in_negotiate_context_offset = 0;
+               uint16_t in_negotiate_context_count = 0;
+               DATA_BLOB in_negotiate_context_blob = data_blob_null;
+               size_t ofs;
 
-               dialect = SVAL(indyn, c*2);
-               if (dialect == SMB2_DIALECT_REVISION_222) {
-                       protocol = PROTOCOL_SMB2_22;
-                       break;
-               }
-       }
+               in_negotiate_context_offset = IVAL(inbody, 0x1C);
+               in_negotiate_context_count = SVAL(inbody, 0x20);
 
-       for (c=0; protocol == PROTOCOL_NONE && c < dialect_count; c++) {
-               if (lp_srv_maxprotocol() < PROTOCOL_SMB2_10) {
-                       break;
-               }
-               if (lp_srv_minprotocol() > PROTOCOL_SMB2_10) {
-                       break;
+               ofs = SMB2_HDR_BODY;
+               ofs += SMBD_SMB2_IN_BODY_LEN(req);
+               ofs += expected_dyn_size;
+               if ((ofs % 8) != 0) {
+                       ofs += 8 - (ofs % 8);
                }
 
-               dialect = SVAL(indyn, c*2);
-               if (dialect == SMB2_DIALECT_REVISION_210) {
-                       protocol = PROTOCOL_SMB2_10;
-                       break;
+               if (in_negotiate_context_offset != ofs) {
+                       return smbd_smb2_request_error(req,
+                                       NT_STATUS_INVALID_PARAMETER);
                }
-       }
 
-       for (c=0; protocol == PROTOCOL_NONE && c < dialect_count; c++) {
-               if (lp_srv_maxprotocol() < PROTOCOL_SMB2_02) {
-                       break;
-               }
-               if (lp_srv_minprotocol() > PROTOCOL_SMB2_02) {
-                       break;
-               }
+               ofs -= SMB2_HDR_BODY;
+               ofs -= SMBD_SMB2_IN_BODY_LEN(req);
 
-               dialect = SVAL(indyn, c*2);
-               if (dialect == SMB2_DIALECT_REVISION_202) {
-                       protocol = PROTOCOL_SMB2_02;
-                       break;
+               if (SMBD_SMB2_IN_DYN_LEN(req) < ofs) {
+                       return smbd_smb2_request_error(req,
+                                       NT_STATUS_INVALID_PARAMETER);
                }
-       }
 
-       for (c=0; protocol == PROTOCOL_NONE && c < dialect_count; c++) {
-               if (lp_srv_maxprotocol() < PROTOCOL_SMB2_10) {
-                       break;
+               in_negotiate_context_blob = data_blob_const(indyn,
+                                               SMBD_SMB2_IN_DYN_LEN(req));
+
+               in_negotiate_context_blob.data += ofs;
+               in_negotiate_context_blob.length -= ofs;
+
+               status = smb2_negotiate_context_parse(req,
+                                       in_negotiate_context_blob, &in_c);
+               if (!NT_STATUS_IS_OK(status)) {
+                       return smbd_smb2_request_error(req, status);
                }
 
-               dialect = SVAL(indyn, c*2);
-               if (dialect == SMB2_DIALECT_REVISION_2FF) {
-                       if (req->sconn->smb2.negprot_2ff) {
-                               req->sconn->smb2.negprot_2ff = false;
-                               protocol = PROTOCOL_SMB2_10;
-                               break;
-                       }
+               if (in_negotiate_context_count != in_c.num_contexts) {
+                       return smbd_smb2_request_error(req,
+                                       NT_STATUS_INVALID_PARAMETER);
                }
        }
 
-       if (protocol == PROTOCOL_NONE) {
-               return smbd_smb2_request_error(req, NT_STATUS_NOT_SUPPORTED);
+       switch (get_remote_arch()) {
+       case RA_VISTA:
+       case RA_SAMBA:
+       case RA_CIFSFS:
+       case RA_OSX:
+               break;
+       default:
+               set_remote_arch(RA_VISTA);
+               break;
        }
 
-       if (get_remote_arch() != RA_SAMBA) {
-               set_remote_arch(RA_VISTA);
+       fstr_sprintf(remote_proto, "SMB%X_%02X",
+                    (dialect >> 8) & 0xFF, dialect & 0xFF);
+
+       reload_services(req->sconn, conn_snum_used, true);
+       DEBUG(3,("Selected protocol %s\n", remote_proto));
+
+       in_preauth = smb2_negotiate_context_find(&in_c,
+                                       SMB2_PREAUTH_INTEGRITY_CAPABILITIES);
+       if (protocol >= PROTOCOL_SMB3_10 && in_preauth == NULL) {
+               return smbd_smb2_request_error(req, NT_STATUS_INVALID_PARAMETER);
        }
+       in_cipher = smb2_negotiate_context_find(&in_c,
+                                       SMB2_ENCRYPTION_CAPABILITIES);
 
        /* negprot_spnego() returns a the server guid in the first 16 bytes */
-       negprot_spnego_blob = negprot_spnego(req, req->sconn);
+       negprot_spnego_blob = negprot_spnego(req, xconn);
        if (negprot_spnego_blob.data == NULL) {
                return smbd_smb2_request_error(req, NT_STATUS_NO_MEMORY);
        }
@@ -245,7 +296,13 @@ NTSTATUS smbd_smb2_request_process_negprot(struct smbd_smb2_request *req)
        }
 
        security_mode = SMB2_NEGOTIATE_SIGNING_ENABLED;
-       if (lp_server_signing() == SMB_SIGNING_REQUIRED) {
+       /*
+        * We use xconn->smb1.signing_state as that's already present
+        * and used lpcfg_server_signing_allowed() to get the correct
+        * defaults, e.g. signing_required for an ad_dc.
+        */
+       signing_required = smb_signing_is_mandatory(xconn->smb1.signing_state);
+       if (signing_required) {
                security_mode |= SMB2_NEGOTIATE_SIGNING_REQUIRED;
        }
 
@@ -254,12 +311,18 @@ NTSTATUS smbd_smb2_request_process_negprot(struct smbd_smb2_request *req)
                capabilities |= SMB2_CAP_DFS;
        }
 
-       if ((protocol >= PROTOCOL_SMB2_24) &&
-           (lp_smb_encrypt(-1) != SMB_SIGNING_OFF))
+       if (protocol >= PROTOCOL_SMB2_10 &&
+           lp_smb2_leases() &&
+           lp_oplocks(GLOBAL_SECTION_SNUM) &&
+           !lp_kernel_oplocks(GLOBAL_SECTION_SNUM))
        {
-               if (in_capabilities & SMB2_CAP_ENCRYPTION) {
-                       capabilities |= SMB2_CAP_ENCRYPTION;
-               }
+               capabilities |= SMB2_CAP_LEASING;
+       }
+
+       if ((protocol >= PROTOCOL_SMB2_24) &&
+           (lp_smb_encrypt(-1) != SMB_SIGNING_OFF) &&
+           (in_capabilities & SMB2_CAP_ENCRYPTION)) {
+               capabilities |= SMB2_CAP_ENCRYPTION;
        }
 
        /*
@@ -278,16 +341,21 @@ NTSTATUS smbd_smb2_request_process_negprot(struct smbd_smb2_request *req)
                /* largeMTU is not supported over NBT (tcp port 139) */
                if (p != NBT_SMB_PORT) {
                        capabilities |= SMB2_CAP_LARGE_MTU;
-                       req->sconn->smb2.supports_multicredit = true;
-
-                       /* SMB >= 2.1 has 1 MB of allowed size */
-                       max_limit = 0x100000; /* 1MB */
+                       xconn->smb2.credits.multicredit = true;
+
+                       /*
+                        * We allow up to almost 16MB.
+                        *
+                        * The maximum PDU size is 0xFFFFFF (16776960)
+                        * and we need some space for the header.
+                        */
+                       max_limit = 0xFFFF00;
                }
        }
 
        /*
-        * the defaults are 1MB, but we'll limit this to max_limit based on
-        * the dialect (64kb for SMB2.0, 1MB for SMB2.1 with LargeMTU)
+        * the defaults are 8MB, but we'll limit this to max_limit based on
+        * the dialect (64kb for SMB 2.0, 8MB for SMB >= 2.1 with LargeMTU)
         *
         * user configured values exceeding the limits will be overwritten,
         * only smaller values will be accepted
@@ -297,6 +365,149 @@ NTSTATUS smbd_smb2_request_process_negprot(struct smbd_smb2_request *req)
        max_read = MIN(max_limit, lp_smb2_max_read());
        max_write = MIN(max_limit, lp_smb2_max_write());
 
+       if (in_preauth != NULL) {
+               size_t needed = 4;
+               uint16_t hash_count;
+               uint16_t salt_length;
+               uint16_t selected_preauth = 0;
+               const uint8_t *p;
+               uint8_t buf[38];
+               DATA_BLOB b;
+               size_t i;
+
+               if (in_preauth->data.length < needed) {
+                       return smbd_smb2_request_error(req,
+                                       NT_STATUS_INVALID_PARAMETER);
+               }
+
+               hash_count = SVAL(in_preauth->data.data, 0);
+               salt_length = SVAL(in_preauth->data.data, 2);
+
+               if (hash_count == 0) {
+                       return smbd_smb2_request_error(req,
+                                       NT_STATUS_INVALID_PARAMETER);
+               }
+
+               p = in_preauth->data.data + needed;
+               needed += hash_count * 2;
+               needed += salt_length;
+
+               if (in_preauth->data.length < needed) {
+                       return smbd_smb2_request_error(req,
+                                       NT_STATUS_INVALID_PARAMETER);
+               }
+
+               for (i=0; i < hash_count; i++) {
+                       uint16_t v;
+
+                       v = SVAL(p, 0);
+                       p += 2;
+
+                       if (v == SMB2_PREAUTH_INTEGRITY_SHA512) {
+                               selected_preauth = v;
+                               break;
+                       }
+               }
+
+               if (selected_preauth == 0) {
+                       return smbd_smb2_request_error(req,
+                               NT_STATUS_SMB_NO_PREAUTH_INTEGRITY_HASH_OVERLAP);
+               }
+
+               SSVAL(buf, 0,  1); /* HashAlgorithmCount */
+               SSVAL(buf, 2, 32); /* SaltLength */
+               SSVAL(buf, 4, selected_preauth);
+               generate_random_buffer(buf + 6, 32);
+
+               b = data_blob_const(buf, sizeof(buf));
+               status = smb2_negotiate_context_add(req, &out_c,
+                                       SMB2_PREAUTH_INTEGRITY_CAPABILITIES, b);
+               if (!NT_STATUS_IS_OK(status)) {
+                       return smbd_smb2_request_error(req, status);
+               }
+
+               req->preauth = &req->xconn->smb2.preauth;
+       }
+
+       if (in_cipher != NULL) {
+               size_t needed = 2;
+               uint16_t cipher_count;
+               const uint8_t *p;
+               uint8_t buf[4];
+               DATA_BLOB b;
+               size_t i;
+               bool aes_128_ccm_supported = false;
+               bool aes_128_gcm_supported = false;
+
+               capabilities &= ~SMB2_CAP_ENCRYPTION;
+
+               if (in_cipher->data.length < needed) {
+                       return smbd_smb2_request_error(req,
+                                       NT_STATUS_INVALID_PARAMETER);
+               }
+
+               cipher_count = SVAL(in_cipher->data.data, 0);
+
+               if (cipher_count == 0) {
+                       return smbd_smb2_request_error(req,
+                                       NT_STATUS_INVALID_PARAMETER);
+               }
+
+               p = in_cipher->data.data + needed;
+               needed += cipher_count * 2;
+
+               if (in_cipher->data.length < needed) {
+                       return smbd_smb2_request_error(req,
+                                       NT_STATUS_INVALID_PARAMETER);
+               }
+
+               for (i=0; i < cipher_count; i++) {
+                       uint16_t v;
+
+                       v = SVAL(p, 0);
+                       p += 2;
+
+                       if (v == SMB2_ENCRYPTION_AES128_GCM) {
+                               aes_128_gcm_supported = true;
+                       }
+                       if (v == SMB2_ENCRYPTION_AES128_CCM) {
+                               aes_128_ccm_supported = true;
+                       }
+               }
+
+               /*
+                * For now we preferr CCM because our implementation
+                * is faster than GCM, see bug #11451.
+                */
+               if (aes_128_ccm_supported) {
+                       xconn->smb2.server.cipher = SMB2_ENCRYPTION_AES128_CCM;
+               } else if (aes_128_gcm_supported) {
+                       xconn->smb2.server.cipher = SMB2_ENCRYPTION_AES128_GCM;
+               }
+
+               SSVAL(buf, 0, 1); /* ChiperCount */
+               SSVAL(buf, 2, xconn->smb2.server.cipher);
+
+               b = data_blob_const(buf, sizeof(buf));
+               status = smb2_negotiate_context_add(req, &out_c,
+                                       SMB2_ENCRYPTION_CAPABILITIES, b);
+               if (!NT_STATUS_IS_OK(status)) {
+                       return smbd_smb2_request_error(req, status);
+               }
+       }
+
+       if (capabilities & SMB2_CAP_ENCRYPTION) {
+               xconn->smb2.server.cipher = SMB2_ENCRYPTION_AES128_CCM;
+       }
+
+       if (protocol >= PROTOCOL_SMB2_22 &&
+           xconn->client->server_multi_channel_enabled)
+       {
+               if (in_capabilities & SMB2_CAP_MULTI_CHANNEL) {
+                       capabilities |= SMB2_CAP_MULTI_CHANNEL;
+               }
+       }
+
        security_offset = SMB2_HDR_BODY + 0x40;
 
 #if 1
@@ -308,13 +519,60 @@ NTSTATUS smbd_smb2_request_process_negprot(struct smbd_smb2_request *req)
        security_buffer = data_blob_const(NULL, 0);
 #endif
 
+       if (out_c.num_contexts != 0) {
+               status = smb2_negotiate_context_push(req,
+                                               &out_negotiate_context_blob,
+                                               out_c);
+               if (!NT_STATUS_IS_OK(status)) {
+                       return smbd_smb2_request_error(req, status);
+               }
+       }
+
+       if (out_negotiate_context_blob.length != 0) {
+               static const uint8_t zeros[8];
+               size_t pad = 0;
+               size_t ofs;
+               bool ok;
+
+               outdyn = data_blob_dup_talloc(req, security_buffer);
+               if (outdyn.length != security_buffer.length) {
+                       return smbd_smb2_request_error(req,
+                                               NT_STATUS_NO_MEMORY);
+               }
+
+               ofs = security_offset + security_buffer.length;
+               if ((ofs % 8) != 0) {
+                       pad = 8 - (ofs % 8);
+               }
+               ofs += pad;
+
+               ok = data_blob_append(req, &outdyn, zeros, pad);
+               if (!ok) {
+                       return smbd_smb2_request_error(req,
+                                               NT_STATUS_NO_MEMORY);
+               }
+
+               ok = data_blob_append(req, &outdyn,
+                                     out_negotiate_context_blob.data,
+                                     out_negotiate_context_blob.length);
+               if (!ok) {
+                       return smbd_smb2_request_error(req,
+                                               NT_STATUS_NO_MEMORY);
+               }
+
+               out_negotiate_context_offset = ofs;
+               out_negotiate_context_count = out_c.num_contexts;
+       } else {
+               outdyn = security_buffer;
+       }
+
        out_guid_blob = data_blob_const(negprot_spnego_blob.data, 16);
        status = GUID_from_ndr_blob(&out_guid_blob, &out_guid);
        if (!NT_STATUS_IS_OK(status)) {
                return smbd_smb2_request_error(req, status);
        }
 
-       outbody = data_blob_talloc(req->out.vector, NULL, 0x40);
+       outbody = smbd_smb2_generate_outbody(req, 0x40);
        if (outbody.data == NULL) {
                return smbd_smb2_request_error(req, NT_STATUS_NO_MEMORY);
        }
@@ -323,7 +581,8 @@ NTSTATUS smbd_smb2_request_process_negprot(struct smbd_smb2_request *req)
        SSVAL(outbody.data, 0x02,
              security_mode);                   /* security mode */
        SSVAL(outbody.data, 0x04, dialect);     /* dialect revision */
-       SSVAL(outbody.data, 0x06, 0);           /* reserved */
+       SSVAL(outbody.data, 0x06,
+             out_negotiate_context_count);     /* reserved/NegotiateContextCount */
        memcpy(outbody.data + 0x08,
               out_guid_blob.data, 16); /* server guid */
        SIVAL(outbody.data, 0x18,
@@ -337,45 +596,96 @@ NTSTATUS smbd_smb2_request_process_negprot(struct smbd_smb2_request *req)
              security_offset);                 /* security buffer offset */
        SSVAL(outbody.data, 0x3A,
              security_buffer.length);          /* security buffer length */
-       SIVAL(outbody.data, 0x3C, 0);           /* reserved */
-
-       outdyn = security_buffer;
+       SIVAL(outbody.data, 0x3C,
+             out_negotiate_context_offset);    /* reserved/NegotiateContextOffset */
 
        req->sconn->using_smb2 = true;
 
        if (dialect != SMB2_DIALECT_REVISION_2FF) {
-               struct smbXsrv_connection *conn = req->sconn->conn;
+               struct smbXsrv_client_global0 *global0 = NULL;
 
-               status = smbXsrv_connection_init_tables(conn, protocol);
+               status = smbXsrv_connection_init_tables(xconn, protocol);
                if (!NT_STATUS_IS_OK(status)) {
                        return smbd_smb2_request_error(req, status);
                }
 
-               conn->smb2.client.capabilities = in_capabilities;
-               conn->smb2.client.security_mode = in_security_mode;
-               conn->smb2.client.guid = in_guid;
-               conn->smb2.client.num_dialects = dialect_count;
-               conn->smb2.client.dialects = talloc_array(conn,
-                                                         uint16_t,
-                                                         dialect_count);
-               if (conn->smb2.client.dialects == NULL) {
+               xconn->smb2.client.capabilities = in_capabilities;
+               xconn->smb2.client.security_mode = in_security_mode;
+               xconn->smb2.client.guid = in_guid;
+               xconn->smb2.client.num_dialects = dialect_count;
+               xconn->smb2.client.dialects = talloc_array(xconn,
+                                                          uint16_t,
+                                                          dialect_count);
+               if (xconn->smb2.client.dialects == NULL) {
                        return smbd_smb2_request_error(req, NT_STATUS_NO_MEMORY);
                }
                for (c=0; c < dialect_count; c++) {
-                       conn->smb2.client.dialects[c] = SVAL(indyn, c*2);
+                       xconn->smb2.client.dialects[c] = SVAL(indyn, c*2);
                }
 
-               conn->smb2.server.capabilities = capabilities;
-               conn->smb2.server.security_mode = security_mode;
-               conn->smb2.server.guid = out_guid;
-               conn->smb2.server.dialect = dialect;
-               conn->smb2.server.max_trans = max_trans;
-               conn->smb2.server.max_read  = max_read;
-               conn->smb2.server.max_write = max_write;
+               xconn->smb2.server.capabilities = capabilities;
+               xconn->smb2.server.security_mode = security_mode;
+               xconn->smb2.server.guid = out_guid;
+               xconn->smb2.server.dialect = dialect;
+               xconn->smb2.server.max_trans = max_trans;
+               xconn->smb2.server.max_read  = max_read;
+               xconn->smb2.server.max_write = max_write;
+
+               if (xconn->protocol < PROTOCOL_SMB2_10) {
+                       /*
+                        * SMB2_02 doesn't support client guids
+                        */
+                       return smbd_smb2_request_done(req, outbody, &outdyn);
+               }
 
-               req->sconn->smb2.max_trans = max_trans;
-               req->sconn->smb2.max_read  = max_read;
-               req->sconn->smb2.max_write = max_write;
+               if (!xconn->client->server_multi_channel_enabled) {
+                       /*
+                        * Only deal with the client guid database
+                        * if multi-channel is enabled.
+                        */
+                       return smbd_smb2_request_done(req, outbody, &outdyn);
+               }
+
+               if (xconn->smb2.client.guid_verified) {
+                       /*
+                        * The connection was passed from another
+                        * smbd process.
+                        */
+                       return smbd_smb2_request_done(req, outbody, &outdyn);
+               }
+
+               status = smb2srv_client_lookup_global(xconn->client,
+                                               xconn->smb2.client.guid,
+                                               req, &global0);
+               /*
+                * TODO: check for races...
+                */
+               if (NT_STATUS_EQUAL(status, NT_STATUS_OBJECTID_NOT_FOUND)) {
+                       /*
+                        * This stores the new client information in
+                        * smbXsrv_client_global.tdb
+                        */
+                       xconn->client->global->client_guid =
+                                               xconn->smb2.client.guid;
+                       status = smbXsrv_client_update(xconn->client);
+                       if (!NT_STATUS_IS_OK(status)) {
+                               return status;
+                       }
+
+                       xconn->smb2.client.guid_verified = true;
+               } else if (NT_STATUS_IS_OK(status)) {
+                       status = smb2srv_client_connection_pass(req,
+                                                               global0);
+                       if (!NT_STATUS_IS_OK(status)) {
+                               return smbd_smb2_request_error(req, status);
+                       }
+
+                       smbd_server_connection_terminate(xconn,
+                                                        "passed connection");
+                       return NT_STATUS_OBJECTID_EXISTS;
+               } else {
+                       return smbd_smb2_request_error(req, status);
+               }
        }
 
        return smbd_smb2_request_done(req, outbody, &outdyn);