Browse Source

GSSAPI: Cases where the preferred authentication mechanism is not supported were not handled correctly

Tal Aloni 7 years ago
parent
commit
eb24eafea6
1 changed files with 27 additions and 4 deletions
  1. 27 4
      SMBLibrary/Authentication/GSSAPI/GSSProvider.cs

+ 27 - 4
SMBLibrary/Authentication/GSSAPI/GSSProvider.cs

@@ -49,12 +49,35 @@ namespace SMBLibrary.Authentication.GSSAPI
                 if (spnegoToken is SimpleProtectedNegotiationTokenInit)
                 {
                     SimpleProtectedNegotiationTokenInit tokenInit = (SimpleProtectedNegotiationTokenInit)spnegoToken;
-                    IGSSMechanism mechanism = FindMechanism(tokenInit.MechanismTypeList);
+                    if (tokenInit.MechanismTypeList.Count == 0)
+                    {
+                        return NTStatus.SEC_E_INVALID_TOKEN;
+                    }
+
+                    // RFC 4178: Note that in order to avoid an extra round trip, the first context establishment token
+                    // of the initiator's preferred mechanism SHOULD be embedded in the initial negotiation message.
+                    byte[] preferredMechanism = tokenInit.MechanismTypeList[0];
+                    IGSSMechanism mechanism = FindMechanism(preferredMechanism);
+                    bool isPreferredMechanism = (mechanism != null);
+                    if (!isPreferredMechanism)
+                    {
+                        mechanism = FindMechanism(tokenInit.MechanismTypeList);
+                    }
+
                     if (mechanism != null)
                     {
-                        byte[] mechanismOutput;
-                        NTStatus status = mechanism.AcceptSecurityContext(ref context, tokenInit.MechanismToken, out mechanismOutput);
-                        outputToken = GetSPNEGOTokenResponseBytes(mechanismOutput, status, mechanism.Identifier);
+                        NTStatus status;
+                        if (isPreferredMechanism)
+                        {
+                            byte[] mechanismOutput;
+                            status = mechanism.AcceptSecurityContext(ref context, tokenInit.MechanismToken, out mechanismOutput);
+                            outputToken = GetSPNEGOTokenResponseBytes(mechanismOutput, status, mechanism.Identifier);
+                        }
+                        else
+                        {
+                            status = NTStatus.SEC_I_CONTINUE_NEEDED;
+                            outputToken = GetSPNEGOTokenResponseBytes(null, status, mechanism.Identifier);
+                        }
                         m_contextToMechanism[context] = mechanism;
                         return status;
                     }