Blame view

trunk/src/rtmp/srs_protocol_handshake.cpp 39.5 KB
winlin authored
1 2 3
/*
The MIT License (MIT)
4
Copyright (c) 2013-2014 winlin
winlin authored
5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23

Permission is hereby granted, free of charge, to any person obtaining a copy of
this software and associated documentation files (the "Software"), to deal in
the Software without restriction, including without limitation the rights to
use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
the Software, and to permit persons to whom the Software is furnished to do so,
subject to the following conditions:

The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
*/
24
#include <srs_protocol_handshake.hpp>
winlin authored
25 26 27

#include <time.h>
28
#include <srs_core_autofree.hpp>
29
#include <srs_kernel_error.hpp>
30
#include <srs_kernel_log.hpp>
31
#include <srs_protocol_io.hpp>
32 33
#include <srs_protocol_utility.hpp>
#include <srs_protocol_rtmp.hpp>
34
#include <srs_kernel_stream.hpp>
winlin authored
35
36
#ifdef SRS_AUTO_SSL
37
38
using namespace _srs_internal;
winlin authored
39
40
// for openssl_HMACsha256
winlin authored
41 42
#include <openssl/evp.h>
#include <openssl/hmac.h>
43
// for __openssl_generate_key
winlin authored
44 45
#include <openssl/dh.h>
46
namespace _srs_internal
winlin authored
47
{
48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72
    // 68bytes FMS key which is used to sign the sever packet.
    u_int8_t SrsGenuineFMSKey[] = {
        0x47, 0x65, 0x6e, 0x75, 0x69, 0x6e, 0x65, 0x20,
        0x41, 0x64, 0x6f, 0x62, 0x65, 0x20, 0x46, 0x6c,
        0x61, 0x73, 0x68, 0x20, 0x4d, 0x65, 0x64, 0x69,
        0x61, 0x20, 0x53, 0x65, 0x72, 0x76, 0x65, 0x72,
        0x20, 0x30, 0x30, 0x31, // Genuine Adobe Flash Media Server 001
        0xf0, 0xee, 0xc2, 0x4a, 0x80, 0x68, 0xbe, 0xe8,
        0x2e, 0x00, 0xd0, 0xd1, 0x02, 0x9e, 0x7e, 0x57,
        0x6e, 0xec, 0x5d, 0x2d, 0x29, 0x80, 0x6f, 0xab,
        0x93, 0xb8, 0xe6, 0x36, 0xcf, 0xeb, 0x31, 0xae
    }; // 68
    
    // 62bytes FP key which is used to sign the client packet.
    u_int8_t SrsGenuineFPKey[] = {
        0x47, 0x65, 0x6E, 0x75, 0x69, 0x6E, 0x65, 0x20,
        0x41, 0x64, 0x6F, 0x62, 0x65, 0x20, 0x46, 0x6C,
        0x61, 0x73, 0x68, 0x20, 0x50, 0x6C, 0x61, 0x79,
        0x65, 0x72, 0x20, 0x30, 0x30, 0x31, // Genuine Adobe Flash Player 001
        0xF0, 0xEE, 0xC2, 0x4A, 0x80, 0x68, 0xBE, 0xE8,
        0x2E, 0x00, 0xD0, 0xD1, 0x02, 0x9E, 0x7E, 0x57,
        0x6E, 0xEC, 0x5D, 0x2D, 0x29, 0x80, 0x6F, 0xAB,
        0x93, 0xB8, 0xE6, 0x36, 0xCF, 0xEB, 0x31, 0xAE
    }; // 62
    
73
    int __openssl_HMACsha256(HMAC_CTX* ctx, const void* data, int data_size, void* digest, unsigned int* digest_size) 
74
    {
75
        int ret = ERROR_SUCCESS;
76
        
77 78 79 80
        if (HMAC_Update(ctx, (unsigned char *) data, data_size) < 0) {
            ret = ERROR_OpenSslSha256Update;
            return ret;
        }
81
    
82 83 84 85
        if (HMAC_Final(ctx, (unsigned char *) digest, digest_size) < 0) {
            ret = ERROR_OpenSslSha256Final;
            return ret;
        }
86
        
87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106
        return ret;
    }
    /**
    * sha256 digest algorithm.
    * @param key the sha256 key, NULL to use EVP_Digest, for instance,
    *       hashlib.sha256(data).digest().
    */
    int openssl_HMACsha256(const void* key, int key_size, const void* data, int data_size, void* digest) 
    {
        int ret = ERROR_SUCCESS;
        
        unsigned int digest_size = 0;
        
        unsigned char* __key = (unsigned char*)key;
        unsigned char* __digest = (unsigned char*)digest;
        
        if (key == NULL) {
            // use data to digest.
            // @see ./crypto/sha/sha256t.c
            // @see ./crypto/evp/digest.c
107
            if (EVP_Digest(data, data_size, __digest, &digest_size, EVP_sha256(), NULL) < 0)
108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124
            {
                ret = ERROR_OpenSslSha256EvpDigest;
                return ret;
            }
        } else {
            // use key-data to digest.
            HMAC_CTX ctx;
            
            // @remark, if no key, use EVP_Digest to digest,
            // for instance, in python, hashlib.sha256(data).digest().
            HMAC_CTX_init(&ctx);
            
            if (HMAC_Init_ex(&ctx, __key, key_size, EVP_sha256(), NULL) < 0) {
                ret = ERROR_OpenSslSha256Init;
                return ret;
            }
            
125
            ret = __openssl_HMACsha256(&ctx, data, data_size, __digest, &digest_size);
126 127 128 129 130 131
            HMAC_CTX_cleanup(&ctx);
            
            if (ret != ERROR_SUCCESS) {
                return ret;
            }
        }
132 133
        
        if (digest_size != 32) {
134 135
            ret = ERROR_OpenSslSha256DigestSize;
            return ret;
136 137
        }
        
138
        return ret;
139 140 141 142 143 144 145 146 147 148
    }
    
    #define RFC2409_PRIME_1024 \
            "FFFFFFFFFFFFFFFFC90FDAA22168C234C4C6628B80DC1CD1" \
            "29024E088A67CC74020BBEA63B139B22514A08798E3404DD" \
            "EF9519B3CD3A431B302B0A6DF25F14374FE1356D6D51C245" \
            "E485B576625E7EC6F44C42E9A637ED6B0BFF5CB6F406B7ED" \
            "EE386BFB5A899FA5AE9F24117C4B1FE649286651ECE65381" \
            "FFFFFFFFFFFFFFFF"
    
149 150 151 152
    SrsDH::SrsDH()
    {
        pdh = NULL;
    }
153
    
154 155 156 157 158 159 160 161 162 163 164 165 166
    SrsDH::~SrsDH()
    {
        if (pdh != NULL) {
            if (pdh->p != NULL) {
                BN_free(pdh->p);
                pdh->p = NULL;
            }
            if (pdh->g != NULL) {
                BN_free(pdh->g);
                pdh->g = NULL;
            }
            DH_free(pdh);
            pdh = NULL;
167
        }
168
    }
169
    
170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187
    int SrsDH::initialize(bool ensure_128bytes_public_key)
    {
        int ret = ERROR_SUCCESS;
        
        for (;;) {
            if ((ret = do_initialize()) != ERROR_SUCCESS) {
                return ret;
            }
            
            if (ensure_128bytes_public_key) {
                int32_t key_size = BN_num_bytes(pdh->pub_key);
                if (key_size != 128) {
                    srs_warn("regenerate 128B key, current=%dB", key_size);
                    continue;
                }
            }
            
            break;
188 189
        }
        
190 191
        return ret;
    }
192
    
193
    int SrsDH::copy_public_key(char* pkey, int32_t& pkey_size)
194 195 196 197
    {
        int ret = ERROR_SUCCESS;
        
        // copy public key to bytes.
198
        // sometimes, the key_size is 127, seems ok.
199
        int32_t key_size = BN_num_bytes(pdh->pub_key);
200
        srs_assert(key_size > 0);
201
        
202 203 204
        // maybe the key_size is 127, but dh will write all 128bytes pkey,
        // so, donot need to set/initialize the pkey.
        // @see https://github.com/winlinvip/simple-rtmp-server/issues/165
205 206 207
        key_size = BN_bn2bin(pdh->pub_key, (unsigned char*)pkey);
        srs_assert(key_size > 0);
        
208 209 210 211
        // output the size of public key.
        // @see https://github.com/winlinvip/simple-rtmp-server/issues/165
        srs_assert(key_size <= pkey_size);
        pkey_size = key_size;
212 213
        
        return ret;
214
    }
215
    
216
    int SrsDH::copy_shared_key(const char* ppkey, int32_t ppkey_size, char* skey, int32_t& skey_size)
217 218 219 220
    {
        int ret = ERROR_SUCCESS;
        
        BIGNUM* ppk = NULL;
221
        if ((ppk = BN_bin2bn((const unsigned char*)ppkey, ppkey_size, 0)) == NULL) {
222 223 224 225
            ret = ERROR_OpenSslGetPeerPublicKey;
            return ret;
        }
        
226
        // if failed, donot return, do cleanup, @see ./test/dhtest.c:168
227 228 229
        // maybe the key_size is 127, but dh will write all 128bytes skey,
        // so, donot need to set/initialize the skey.
        // @see https://github.com/winlinvip/simple-rtmp-server/issues/165
230 231 232 233 234 235
        int32_t key_size = DH_compute_key((unsigned char*)skey, ppk, pdh);
        
        if (key_size < ppkey_size) {
            srs_warn("shared key size=%d, ppk_size=%d", key_size, ppkey_size);
        }
        
236
        if (key_size < 0 || key_size > skey_size) {
237
            ret = ERROR_OpenSslComputeSharedKey;
238
        } else {
239
            skey_size = key_size;
240 241 242 243 244 245 246 247
        }
        
        if (ppk) {
            BN_free(ppk);
        }
        
        return ret;
    }
248 249
    
    int SrsDH::do_initialize()
250 251 252
    {
        int ret = ERROR_SUCCESS;
        
253 254 255 256 257
        int32_t bits_count = 1024; 
        
        //1. Create the DH
        if ((pdh = DH_new()) == NULL) {
            ret = ERROR_OpenSslCreateDH; 
258 259
            return ret;
        }
260 261 262 263
    
        //2. Create his internal p and g
        if ((pdh->p = BN_new()) == NULL) {
            ret = ERROR_OpenSslCreateP; 
264 265
            return ret;
        }
266 267 268 269 270 271 272 273 274 275 276 277 278 279
        if ((pdh->g = BN_new()) == NULL) {
            ret = ERROR_OpenSslCreateG; 
            return ret;
        }
    
        //3. initialize p and g, @see ./test/ectest.c:260
        if (!BN_hex2bn(&pdh->p, RFC2409_PRIME_1024)) {
            ret = ERROR_OpenSslParseP1024; 
            return ret;
        }
        // @see ./test/bntest.c:1764
        if (!BN_set_word(pdh->g, 2)) {
            ret = ERROR_OpenSslSetG;
            return ret;
280
        }
281
    
282 283
        // 4. Set the key length
        pdh->length = bits_count;
284
    
285 286 287 288
        // 5. Generate private and public key
        // @see ./test/dhtest.c:152
        if (!DH_generate_key(pdh)) {
            ret = ERROR_OpenSslGenerateDHKeys;
289
            return ret;
290
        }
291
        
292 293
        return ret;
    }
294
    
295
    key_block::key_block()
296
    {
297 298 299 300 301 302 303 304 305 306 307 308
        offset = (int32_t)rand();
        random0 = NULL;
        random1 = NULL;
        
        int valid_offset = calc_valid_offset();
        srs_assert(valid_offset >= 0);
        
        random0_size = valid_offset;
        if (random0_size > 0) {
            random0 = new char[random0_size];
            srs_random_generate(random0, random0_size);
            snprintf(random0, random0_size, "%s", RTMP_SIG_SRS_HANDSHAKE);
309 310
        }
        
311
        srs_random_generate(key, sizeof(key));
312
        
313 314 315 316 317
        random1_size = 764 - valid_offset - 128 - 4;
        if (random1_size > 0) {
            random1 = new char[random1_size];
            srs_random_generate(random1, random1_size);
            snprintf(random1, random1_size, "%s", RTMP_SIG_SRS_HANDSHAKE);
318
        }
319 320
    }
    
321
    key_block::~key_block()
322
    {
323 324
        srs_freep(random0);
        srs_freep(random1);
325 326
    }
    
327
    int key_block::parse(SrsStream* stream)
328 329
    {
        int ret = ERROR_SUCCESS;
330 331 332
        
        // the key must be 764 bytes.
        srs_assert(stream->require(764));
333
    
334 335
        // read the last offset first, 760-763
        stream->skip(764 - sizeof(int32_t));
336
        offset = stream->read_4bytes();
337
        
338 339
        // reset stream to read others.
        stream->skip(-764);
340
        
341 342
        int valid_offset = calc_valid_offset();
        srs_assert(valid_offset >= 0);
343
        
344 345
        random0_size = valid_offset;
        if (random0_size > 0) {
346
            srs_freep(random0);
347 348
            random0 = new char[random0_size];
            stream->read_bytes(random0, random0_size);
349 350
        }
        
351
        stream->read_bytes(key, 128);
352
        
353 354
        random1_size = 764 - valid_offset - 128 - 4;
        if (random1_size > 0) {
355
            srs_freep(random1);
356 357
            random1 = new char[random1_size];
            stream->read_bytes(random1, random1_size);
358 359 360
        }
        
        return ret;
361 362
    }
    
363
    int key_block::calc_valid_offset()
364
    {
365
        int max_offset_size = 764 - 128 - 4;
366
        
367 368 369 370 371 372 373 374
        int valid_offset = 0;
        u_int8_t* pp = (u_int8_t*)&offset;
        valid_offset += *pp++;
        valid_offset += *pp++;
        valid_offset += *pp++;
        valid_offset += *pp++;
    
        return valid_offset % max_offset_size;
375 376
    }
    
377
    digest_block::digest_block()
378
    {
379 380 381 382 383 384 385 386 387 388 389 390
        offset = (int32_t)rand();
        random0 = NULL;
        random1 = NULL;
        
        int valid_offset = calc_valid_offset();
        srs_assert(valid_offset >= 0);
        
        random0_size = valid_offset;
        if (random0_size > 0) {
            random0 = new char[random0_size];
            srs_random_generate(random0, random0_size);
            snprintf(random0, random0_size, "%s", RTMP_SIG_SRS_HANDSHAKE);
391 392
        }
        
393
        srs_random_generate(digest, sizeof(digest));
394
        
395 396 397 398 399
        random1_size = 764 - 4 - valid_offset - 32;
        if (random1_size > 0) {
            random1 = new char[random1_size];
            srs_random_generate(random1, random1_size);
            snprintf(random1, random1_size, "%s", RTMP_SIG_SRS_HANDSHAKE);
400
        }
401
    }
402
    
403
    digest_block::~digest_block()
404
    {
405 406
        srs_freep(random0);
        srs_freep(random1);
407
    }
408
409
    int digest_block::parse(SrsStream* stream)
410 411 412
    {
        int ret = ERROR_SUCCESS;
        
413 414 415
        // the digest must be 764 bytes.
        srs_assert(stream->require(764));
        
416
        offset = stream->read_4bytes();
417
        
418 419
        int valid_offset = calc_valid_offset();
        srs_assert(valid_offset >= 0);
420
        
421 422
        random0_size = valid_offset;
        if (random0_size > 0) {
423
            srs_freep(random0);
424 425
            random0 = new char[random0_size];
            stream->read_bytes(random0, random0_size);
426 427
        }
        
428
        stream->read_bytes(digest, 32);
429
        
430 431
        random1_size = 764 - 4 - valid_offset - 32;
        if (random1_size > 0) {
432
            srs_freep(random1);
433 434
            random1 = new char[random1_size];
            stream->read_bytes(random1, random1_size);
435 436 437
        }
        
        return ret;
438 439
    }
    
440
    int digest_block::calc_valid_offset()
441
    {
442
        int max_offset_size = 764 - 32 - 4;
443
        
444 445 446 447 448 449 450 451
        int valid_offset = 0;
        u_int8_t* pp = (u_int8_t*)&offset;
        valid_offset += *pp++;
        valid_offset += *pp++;
        valid_offset += *pp++;
        valid_offset += *pp++;
    
        return valid_offset % max_offset_size;
452 453
    }
    
454 455 456 457
    c1s1_strategy::c1s1_strategy()
    {
    }
    
458
    c1s1_strategy::~c1s1_strategy()
459 460 461
    {
    }
    
462
    char* c1s1_strategy::get_digest()
463 464 465 466
    {
        return digest.digest;
    }
    
467 468 469 470 471
    char* c1s1_strategy::get_key()
    {
        return key.key;
    }
    
472
    int c1s1_strategy::dump(c1s1* owner, char* _c1s1, int size)
473
    {
474
        srs_assert(size == 1536);
475
        return copy_to(owner, _c1s1, size, true);
476 477
    }
    
478
    int c1s1_strategy::c1_create(c1s1* owner)
479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497
    {
        int ret = ERROR_SUCCESS;
        
        // generate digest
        char* c1_digest = NULL;
        
        if ((ret = calc_c1_digest(owner, c1_digest)) != ERROR_SUCCESS) {
            srs_error("sign c1 error, failed to calc digest. ret=%d", ret);
            return ret;
        }
        
        srs_assert(c1_digest != NULL);
        SrsAutoFree(char, c1_digest);
        
        memcpy(digest.digest, c1_digest, 32);
        
        return ret;
    }
    
498
    int c1s1_strategy::c1_validate_digest(c1s1* owner, bool& is_valid)
499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516
    {
        int ret = ERROR_SUCCESS;
        
        char* c1_digest = NULL;
        
        if ((ret = calc_c1_digest(owner, c1_digest)) != ERROR_SUCCESS) {
            srs_error("validate c1 error, failed to calc digest. ret=%d", ret);
            return ret;
        }
        
        srs_assert(c1_digest != NULL);
        SrsAutoFree(char, c1_digest);
        
        is_valid = srs_bytes_equals(digest.digest, c1_digest, 32);
        
        return ret;
    }
    
517
    int c1s1_strategy::s1_create(c1s1* owner, c1s1* c1)
518 519 520 521 522 523 524 525 526 527 528 529 530
    {
        int ret = ERROR_SUCCESS;

        SrsDH dh;
        
        // ensure generate 128bytes public key.
        if ((ret = dh.initialize(true)) != ERROR_SUCCESS) {
            return ret;
        }
        
        // directly generate the public key.
        // @see: https://github.com/winlinvip/simple-rtmp-server/issues/148
        int pkey_size = 128;
531
        if ((ret = dh.copy_shared_key(c1->get_key(), 128, key.key, pkey_size)) != ERROR_SUCCESS) {
532 533 534
            srs_error("calc s1 key failed. ret=%d", ret);
            return ret;
        }
535 536 537 538 539

        // altough the public key is always 128bytes, but the share key maybe not.
        // we just ignore the actual key size, but if need to use the key, must use the actual size.
        // TODO: FIXME: use the actual key size.
        //srs_assert(pkey_size == 128);
540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557
        srs_verbose("calc s1 key success.");
            
        char* s1_digest = NULL;
        if ((ret = calc_s1_digest(owner, s1_digest))  != ERROR_SUCCESS) {
            srs_error("calc s1 digest failed. ret=%d", ret);
            return ret;
        }
        srs_verbose("calc s1 digest success.");
        
        srs_assert(s1_digest != NULL);
        SrsAutoFree(char, s1_digest);
        
        memcpy(digest.digest, s1_digest, 32);
        srs_verbose("copy s1 key success.");
        
        return ret;
    }
    
558
    int c1s1_strategy::s1_validate_digest(c1s1* owner, bool& is_valid)
559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576
    {
        int ret = ERROR_SUCCESS;
        
        char* s1_digest = NULL;
        
        if ((ret = calc_s1_digest(owner, s1_digest)) != ERROR_SUCCESS) {
            srs_error("validate s1 error, failed to calc digest. ret=%d", ret);
            return ret;
        }
        
        srs_assert(s1_digest != NULL);
        SrsAutoFree(char, s1_digest);
        
        is_valid = srs_bytes_equals(digest.digest, s1_digest, 32);
        
        return ret;
    }
    
577
    int c1s1_strategy::calc_c1_digest(c1s1* owner, char*& c1_digest)
578 579 580
    {
        int ret = ERROR_SUCCESS;
581 582 583 584 585 586 587 588
        /**
        * c1s1 is splited by digest:
        *     c1s1-part1: n bytes (time, version, key and digest-part1).
        *     digest-data: 32bytes
        *     c1s1-part2: (1536-n-32)bytes (digest-part2)
        * @return a new allocated bytes, user must free it.
        */
        char* c1s1_joined_bytes = new char[1536 -32];
589
        SrsAutoFree(char, c1s1_joined_bytes);
590 591 592
        if ((ret = copy_to(owner, c1s1_joined_bytes, 1536 - 32, false)) != ERROR_SUCCESS) {
            return ret;
        }
593 594 595 596 597 598 599 600 601 602 603 604
        
        c1_digest = new char[__SRS_OpensslHashSize];
        if ((ret = openssl_HMACsha256(SrsGenuineFPKey, 30, c1s1_joined_bytes, 1536 - 32, c1_digest)) != ERROR_SUCCESS) {
            srs_freep(c1_digest);
            srs_error("calc digest for c1 failed. ret=%d", ret);
            return ret;
        }
        srs_verbose("digest calculated for c1");
        
        return ret;
    }
    
605
    int c1s1_strategy::calc_s1_digest(c1s1* owner, char*& s1_digest)
606 607
    {
        int ret = ERROR_SUCCESS;
608 609 610 611 612 613 614 615 616

        /**
        * c1s1 is splited by digest:
        *     c1s1-part1: n bytes (time, version, key and digest-part1).
        *     digest-data: 32bytes
        *     c1s1-part2: (1536-n-32)bytes (digest-part2)
        * @return a new allocated bytes, user must free it.
        */
        char* c1s1_joined_bytes = new char[1536 -32];
617
        SrsAutoFree(char, c1s1_joined_bytes);
618 619 620
        if ((ret = copy_to(owner, c1s1_joined_bytes, 1536 - 32, false)) != ERROR_SUCCESS) {
            return ret;
        }
621 622 623 624 625 626 627 628 629 630 631 632
        
        s1_digest = new char[__SRS_OpensslHashSize];
        if ((ret = openssl_HMACsha256(SrsGenuineFMSKey, 36, c1s1_joined_bytes, 1536 - 32, s1_digest)) != ERROR_SUCCESS) {
            srs_freep(s1_digest);
            srs_error("calc digest for s1 failed. ret=%d", ret);
            return ret;
        }
        srs_verbose("digest calculated for s1");

        return ret;
    }
    
633
    void c1s1_strategy::copy_time_version(SrsStream* stream, c1s1* owner)
634
    {
635 636
        srs_assert(stream->require(8));
        
637
        // 4bytes time
638 639
        stream->write_4bytes(owner->time);
640
        // 4bytes version
641
        stream->write_4bytes(owner->version);
642
    }
643
    void c1s1_strategy::copy_key(SrsStream* stream)
644
    {
645 646 647 648 649 650
        srs_assert(key.random0_size >= 0);
        srs_assert(key.random1_size >= 0);
        
        int total = key.random0_size + 128 + key.random1_size + 4;
        srs_assert(stream->require(total));
        
651 652
        // 764bytes key block
        if (key.random0_size > 0) {
653
            stream->write_bytes(key.random0, key.random0_size);
654 655
        }
        
656
        stream->write_bytes(key.key, 128);
657 658
        
        if (key.random1_size > 0) {
659
            stream->write_bytes(key.random1, key.random1_size);
660 661
        }
        
662
        stream->write_4bytes(key.offset);
663
    }
664
    void c1s1_strategy::copy_digest(SrsStream* stream, bool with_digest)
665
    {
666 667 668 669 670 671 672 673 674
        srs_assert(key.random0_size >= 0);
        srs_assert(key.random1_size >= 0);
        
        int total = 4 + digest.random0_size + digest.random1_size;
        if (with_digest) {
            total += 32;
        }
        srs_assert(stream->require(total));
        
675 676
        // 732bytes digest block without the 32bytes digest-data
        // nbytes digest block part1
677
        stream->write_4bytes(digest.offset);
678 679 680
        
        // digest random padding.
        if (digest.random0_size > 0) {
681
            stream->write_bytes(digest.random0, digest.random0_size);
682 683 684 685
        }
        
        // digest
        if (with_digest) {
686
            stream->write_bytes(digest.digest, 32);
687 688 689 690
        }
        
        // nbytes digest block part2
        if (digest.random1_size > 0) {
691
            stream->write_bytes(digest.random1, digest.random1_size);
692
        }
693 694
    }
    
695
    c1s1_strategy_schema0::c1s1_strategy_schema0()
696 697 698
    {
    }
    
699
    c1s1_strategy_schema0::~c1s1_strategy_schema0()
700 701 702
    {
    }
    
703
    srs_schema_type c1s1_strategy_schema0::schema()
704
    {
705
        return srs_schema0;
706 707
    }
    
708
    int c1s1_strategy_schema0::parse(char* _c1s1, int size)
709 710 711
    {
        int ret = ERROR_SUCCESS;
        
712 713 714 715 716 717 718 719 720
        srs_assert(size == 1536);
        
        SrsStream stream;
        
        if ((ret = stream.initialize(_c1s1 + 8, 764)) != ERROR_SUCCESS) {
            return ret;
        }
        
        if ((ret = key.parse(&stream)) != ERROR_SUCCESS) {
721
            srs_error("parse the c1 key failed. ret=%d", ret);
722 723 724
            return ret;
        }
        
725 726 727 728 729
        if ((ret = stream.initialize(_c1s1 + 8 + 764, 764)) != ERROR_SUCCESS) {
            return ret;
        }

        if ((ret = digest.parse(&stream)) != ERROR_SUCCESS) {
730
            srs_error("parse the c1 digest failed. ret=%d", ret);
731 732 733
            return ret;
        }
        
734
        srs_verbose("parse c1 key-digest success");
735 736 737 738
        
        return ret;
    }
    
739
    int c1s1_strategy_schema0::copy_to(c1s1* owner, char* bytes, int size, bool with_digest)
740
    {
741
        int ret = ERROR_SUCCESS;
742
        
743 744 745 746 747 748
        if (with_digest) {
            srs_assert(size == 1536);
        } else {
            srs_assert(size == 1504);
        }
        
749 750
        SrsStream stream;
        
751 752
        if ((ret = stream.initialize(bytes, size)) != ERROR_SUCCESS) {
            return ret;
753
        }
754 755 756 757 758 759
        
        copy_time_version(&stream, owner);
        copy_key(&stream);
        copy_digest(&stream, with_digest);
        
        srs_assert(stream.empty());
760 761
        
        return ret;
762 763
    }
    
764
    c1s1_strategy_schema1::c1s1_strategy_schema1()
765 766 767
    {
    }
    
768
    c1s1_strategy_schema1::~c1s1_strategy_schema1()
769 770 771
    {
    }
    
772
    srs_schema_type c1s1_strategy_schema1::schema()
773
    {
774
        return srs_schema1;
775 776
    }
    
777
    int c1s1_strategy_schema1::parse(char* _c1s1, int size)
778 779 780
    {
        int ret = ERROR_SUCCESS;
        
781 782 783 784 785 786 787 788 789
        srs_assert(size == 1536);
        
        SrsStream stream;
        
        if ((ret = stream.initialize(_c1s1 + 8, 764)) != ERROR_SUCCESS) {
            return ret;
        }

        if ((ret = digest.parse(&stream)) != ERROR_SUCCESS) {
790 791 792
            srs_error("parse the c1 digest failed. ret=%d", ret);
            return ret;
        }
793
        
794 795 796 797 798
        if ((ret = stream.initialize(_c1s1 + 8 + 764, 764)) != ERROR_SUCCESS) {
            return ret;
        }
        
        if ((ret = key.parse(&stream)) != ERROR_SUCCESS) {
799
            srs_error("parse the c1 key failed. ret=%d", ret);
800 801
            return ret;
        }
802 803
        
        srs_verbose("parse c1 digest-key success");
804 805 806 807
        
        return ret;
    }
    
808
    int c1s1_strategy_schema1::copy_to(c1s1* owner, char* bytes, int size, bool with_digest)
809
    {
810
        int ret = ERROR_SUCCESS;
811
        
812 813 814 815 816 817
        if (with_digest) {
            srs_assert(size == 1536);
        } else {
            srs_assert(size == 1504);
        }
        
818 819
        SrsStream stream;
        
820 821
        if ((ret = stream.initialize(bytes, size)) != ERROR_SUCCESS) {
            return ret;
822
        }
823 824 825 826 827 828
        
        copy_time_version(&stream, owner);
        copy_digest(&stream, with_digest);
        copy_key(&stream);
        
        srs_assert(stream.empty());
829 830
        
        return ret;
831 832
    }
    
833 834
    c1s1::c1s1()
    {
835
        payload = NULL;
836
    }
837 838
    c1s1::~c1s1()
    {
839 840 841 842 843 844 845
        srs_freep(payload);
    }
    
    srs_schema_type c1s1::schema()
    {
        srs_assert(payload != NULL);
        return payload->schema();
846 847
    }
    
848 849
    char* c1s1::get_digest()
    {
850 851
        srs_assert(payload != NULL);
        return payload->get_digest();
852 853
    }
    
854 855 856 857 858 859
    char* c1s1::get_key()
    {
        srs_assert(payload != NULL);
        return payload->get_key();
    }
    
860
    int c1s1::dump(char* _c1s1, int size)
861
    {
862
        srs_assert(size == 1536);
863
        srs_assert(payload != NULL);
864
        return payload->dump(this, _c1s1, size);
865 866
    }
    
867
    int c1s1::parse(char* _c1s1, int size, srs_schema_type schema)
868 869 870
    {
        int ret = ERROR_SUCCESS;
        
871 872
        srs_assert(size == 1536);
        
873
        if (schema != srs_schema0 && schema != srs_schema1) {
874
            ret = ERROR_RTMP_CH_SCHEMA;
875
            srs_error("parse c1 failed. invalid schema=%d, ret=%d", schema, ret);
876 877 878
            return ret;
        }
        
879 880 881 882 883 884 885 886
        SrsStream stream;
        
        if ((ret = stream.initialize(_c1s1, size)) != ERROR_SUCCESS) {
            return ret;
        }
        
        time = stream.read_4bytes();
        version = stream.read_4bytes(); // client c1 version
887
        
888 889 890
        srs_freep(payload);
        if (schema == srs_schema0) {
            payload = new c1s1_strategy_schema0();
891
        } else {
892
            payload = new c1s1_strategy_schema1();
893
        }
894
895
        return payload->parse(_c1s1, size);
896 897
    }
    
898
    int c1s1::c1_create(srs_schema_type schema)
899 900 901
    {
        int ret = ERROR_SUCCESS;
        
902
        if (schema != srs_schema0 && schema != srs_schema1) {
903
            ret = ERROR_RTMP_CH_SCHEMA;
904
            srs_error("create c1 failed. invalid schema=%d, ret=%d", schema, ret);
905 906 907
            return ret;
        }
        
908
        // client c1 time and version
909
        time = ::time(NULL);
910
        version = 0x80000702; // client c1 version
911
912
        // generate signature by schema
913
        srs_freep(payload);
914
        if (schema == srs_schema0) {
915
            payload = new c1s1_strategy_schema0();
916
        } else {
917
            payload = new c1s1_strategy_schema1();
918 919
        }
        
920
        return payload->c1_create(this);
921 922
    }
    
923 924 925
    int c1s1::c1_validate_digest(bool& is_valid)
    {
        is_valid = false;
926 927
        srs_assert(payload);
        return payload->c1_validate_digest(this, is_valid);
928 929
    }
    
930 931 932 933
    int c1s1::s1_create(c1s1* c1)
    {
        int ret = ERROR_SUCCESS;
        
934
        if (c1->schema() != srs_schema0 && c1->schema() != srs_schema1) {
935
            ret = ERROR_RTMP_CH_SCHEMA;
936
            srs_error("create s1 failed. invalid schema=%d, ret=%d", c1->schema(), ret);
937 938
            return ret;
        }
939 940 941 942
        
        time = ::time(NULL);
        version = 0x01000504; // server s1 version
        
943 944 945
        srs_freep(payload);
        if (c1->schema() == srs_schema0) {
            payload = new c1s1_strategy_schema0();
946
        } else {
947
            payload = new c1s1_strategy_schema1();
948 949
        }
        
950
        return payload->s1_create(this, c1);
951
    }
952 953 954 955 956 957 958 959 960 961 962 963 964 965 966 967 968 969 970 971 972 973 974
    
    int c1s1::s1_validate_digest(bool& is_valid)
    {
        is_valid = false;
        srs_assert(payload);
        return payload->s1_validate_digest(this, is_valid);
    }
    
    c2s2::c2s2()
    {
        srs_random_generate(random, 1504);
        
        int size = snprintf(random, 1504, "%s", RTMP_SIG_SRS_HANDSHAKE);
        srs_assert(++size < 1504);
        snprintf(random + 1504 - size, size, "%s", RTMP_SIG_SRS_HANDSHAKE);
        
        srs_random_generate(digest, 32);
    }
    
    c2s2::~c2s2()
    {
    }
    
975
    int c2s2::dump(char* _c2s2, int size)
976
    {
977 978
        srs_assert(size == 1536);
        
979 980
        memcpy(_c2s2, random, 1504);
        memcpy(_c2s2 + 1504, digest, 32);
981 982
        
        return ERROR_SUCCESS;
983 984
    }
    
985
    int c2s2::parse(char* _c2s2, int size)
986
    {
987 988
        srs_assert(size == 1536);
        
989 990
        memcpy(random, _c2s2, 1504);
        memcpy(digest, _c2s2 + 1504, 32);
991 992
        
        return ERROR_SUCCESS;
993 994 995 996 997 998 999 1000 1001 1002 1003 1004 1005 1006 1007 1008 1009 1010 1011 1012 1013 1014 1015 1016 1017 1018 1019 1020 1021 1022 1023 1024 1025 1026 1027 1028 1029 1030 1031 1032 1033 1034 1035 1036 1037 1038 1039 1040 1041 1042 1043 1044 1045 1046 1047 1048 1049 1050 1051 1052 1053 1054 1055 1056 1057 1058 1059 1060 1061 1062 1063 1064 1065 1066 1067 1068 1069 1070 1071 1072 1073 1074 1075 1076 1077 1078 1079 1080 1081 1082 1083 1084 1085 1086 1087
    }
    
    int c2s2::c2_create(c1s1* s1)
    {
        int ret = ERROR_SUCCESS;
        
        char temp_key[__SRS_OpensslHashSize];
        if ((ret = openssl_HMACsha256(SrsGenuineFPKey, 62, s1->get_digest(), 32, temp_key)) != ERROR_SUCCESS) {
            srs_error("create c2 temp key failed. ret=%d", ret);
            return ret;
        }
        srs_verbose("generate c2 temp key success.");
        
        char _digest[__SRS_OpensslHashSize];
        if ((ret = openssl_HMACsha256(temp_key, 32, random, 1504, _digest)) != ERROR_SUCCESS) {
            srs_error("create c2 digest failed. ret=%d", ret);
            return ret;
        }
        srs_verbose("generate c2 digest success.");
        
        memcpy(digest, _digest, 32);
        
        return ret;
    }
    
    int c2s2::c2_validate(c1s1* s1, bool& is_valid)
    {
        is_valid = false;
        int ret = ERROR_SUCCESS;
        
        char temp_key[__SRS_OpensslHashSize];
        if ((ret = openssl_HMACsha256(SrsGenuineFPKey, 62, s1->get_digest(), 32, temp_key)) != ERROR_SUCCESS) {
            srs_error("create c2 temp key failed. ret=%d", ret);
            return ret;
        }
        srs_verbose("generate c2 temp key success.");
        
        char _digest[__SRS_OpensslHashSize];
        if ((ret = openssl_HMACsha256(temp_key, 32, random, 1504, _digest)) != ERROR_SUCCESS) {
            srs_error("create c2 digest failed. ret=%d", ret);
            return ret;
        }
        srs_verbose("generate c2 digest success.");
        
        is_valid = srs_bytes_equals(digest, _digest, 32);
        
        return ret;
    }
    
    int c2s2::s2_create(c1s1* c1)
    {
        int ret = ERROR_SUCCESS;
        
        char temp_key[__SRS_OpensslHashSize];
        if ((ret = openssl_HMACsha256(SrsGenuineFMSKey, 68, c1->get_digest(), 32, temp_key)) != ERROR_SUCCESS) {
            srs_error("create s2 temp key failed. ret=%d", ret);
            return ret;
        }
        srs_verbose("generate s2 temp key success.");
        
        char _digest[__SRS_OpensslHashSize];
        if ((ret = openssl_HMACsha256(temp_key, 32, random, 1504, _digest)) != ERROR_SUCCESS) {
            srs_error("create s2 digest failed. ret=%d", ret);
            return ret;
        }
        srs_verbose("generate s2 digest success.");
        
        memcpy(digest, _digest, 32);
        
        return ret;
    }
    
    int c2s2::s2_validate(c1s1* c1, bool& is_valid)
    {
        is_valid = false;
        int ret = ERROR_SUCCESS;
        
        char temp_key[__SRS_OpensslHashSize];
        if ((ret = openssl_HMACsha256(SrsGenuineFMSKey, 68, c1->get_digest(), 32, temp_key)) != ERROR_SUCCESS) {
            srs_error("create s2 temp key failed. ret=%d", ret);
            return ret;
        }
        srs_verbose("generate s2 temp key success.");
        
        char _digest[__SRS_OpensslHashSize];
        if ((ret = openssl_HMACsha256(temp_key, 32, random, 1504, _digest)) != ERROR_SUCCESS) {
            srs_error("create s2 digest failed. ret=%d", ret);
            return ret;
        }
        srs_verbose("generate s2 digest success.");
        
        is_valid = srs_bytes_equals(digest, _digest, 32);
        
        return ret;
    }
winlin authored
1088 1089
}
1090 1091
#endif
winlin authored
1092 1093 1094 1095 1096 1097 1098 1099
SrsSimpleHandshake::SrsSimpleHandshake()
{
}

SrsSimpleHandshake::~SrsSimpleHandshake()
{
}
1100
int SrsSimpleHandshake::handshake_with_client(SrsHandshakeBytes* hs_bytes, ISrsProtocolReaderWriter* io)
winlin authored
1101
{
1102 1103
    int ret = ERROR_SUCCESS;
    
winlin authored
1104 1105
    ssize_t nsize;
    
1106
    if ((ret = hs_bytes->read_c0c1(io)) != ERROR_SUCCESS) {
winlin authored
1107 1108 1109
        return ret;
    }
1110
    // plain text required.
1111
    if (hs_bytes->c0c1[0] != 0x03) {
1112 1113 1114 1115
        ret = ERROR_RTMP_PLAIN_REQUIRED;
        srs_warn("only support rtmp plain text. ret=%d", ret);
        return ret;
    }
winlin authored
1116 1117
    srs_verbose("check c0 success, required plain text.");
    
1118
    if ((ret = hs_bytes->create_s0s1s2(hs_bytes->c0c1 + 1)) != ERROR_SUCCESS) {
1119
        return ret;
1120 1121
    }
    
1122
    if ((ret = io->write(hs_bytes->s0s1s2, 3073, &nsize)) != ERROR_SUCCESS) {
winlin authored
1123 1124 1125 1126 1127
        srs_warn("simple handshake send s0s1s2 failed. ret=%d", ret);
        return ret;
    }
    srs_verbose("simple handshake send s0s1s2 success.");
    
1128
    if ((ret = hs_bytes->read_c2(io)) != ERROR_SUCCESS) {
winlin authored
1129 1130 1131
        return ret;
    }
    
winlin authored
1132
    srs_trace("simple handshake success.");
winlin authored
1133
    
1134
    return ret;
winlin authored
1135 1136
}
1137
int SrsSimpleHandshake::handshake_with_server(SrsHandshakeBytes* hs_bytes, ISrsProtocolReaderWriter* io)
winlin authored
1138
{
1139
    int ret = ERROR_SUCCESS;
winlin authored
1140 1141 1142
    
    ssize_t nsize;
    
1143 1144 1145 1146
    // simple handshake
    if ((ret = hs_bytes->create_c0c1()) != ERROR_SUCCESS) {
        return ret;
    }
1147
    
1148
    if ((ret = io->write(hs_bytes->c0c1, 1537, &nsize)) != ERROR_SUCCESS) {
winlin authored
1149 1150 1151 1152
        srs_warn("write c0c1 failed. ret=%d", ret);
        return ret;
    }
    srs_verbose("write c0c1 success.");
1153
    
1154
    if ((ret = hs_bytes->read_s0s1s2(io)) != ERROR_SUCCESS) {
winlin authored
1155 1156 1157
        return ret;
    }
    
1158
    // plain text required.
1159
    if (hs_bytes->s0s1s2[0] != 0x03) {
winlin authored
1160 1161 1162 1163 1164
        ret = ERROR_RTMP_HANDSHAKE;
        srs_warn("handshake failed, plain text required. ret=%d", ret);
        return ret;
    }
    
1165 1166 1167 1168
    if ((ret = hs_bytes->create_c2()) != ERROR_SUCCESS) {
        return ret;
    }
    if ((ret = io->write(hs_bytes->c2, 1536, &nsize)) != ERROR_SUCCESS) {
winlin authored
1169 1170 1171 1172 1173
        srs_warn("simple handshake write c2 failed. ret=%d", ret);
        return ret;
    }
    srs_verbose("simple handshake write c2 success.");
    
winlin authored
1174
    srs_trace("simple handshake success.");
winlin authored
1175
    
1176
    return ret;
winlin authored
1177 1178
}
winlin authored
1179 1180 1181 1182 1183 1184 1185 1186
SrsComplexHandshake::SrsComplexHandshake()
{
}

SrsComplexHandshake::~SrsComplexHandshake()
{
}
1187
#ifndef SRS_AUTO_SSL
1188
int SrsComplexHandshake::handshake_with_client(SrsHandshakeBytes* /*hs_bytes*/, ISrsProtocolReaderWriter* /*io*/)
1189
{
winlin authored
1190
    srs_trace("directly use simple handshake for ssl disabled.");
1191
    return ERROR_RTMP_TRY_SIMPLE_HS;
1192 1193
}
#else
1194
int SrsComplexHandshake::handshake_with_client(SrsHandshakeBytes* hs_bytes, ISrsProtocolReaderWriter* io)
winlin authored
1195
{
1196
    int ret = ERROR_SUCCESS;
1197
winlin authored
1198
    ssize_t nsize;
1199
    
1200 1201
    if ((ret = hs_bytes->read_c0c1(io)) != ERROR_SUCCESS) {
        return ret;
1202 1203 1204 1205 1206
    }
    
    // decode c1
    c1s1 c1;
    // try schema0.
1207
    // @remark, use schema0 to make flash player happy.
1208
    if ((ret = c1.parse(hs_bytes->c0c1 + 1, 1536, srs_schema0)) != ERROR_SUCCESS) {
1209 1210 1211 1212 1213 1214
        srs_error("parse c1 schema%d error. ret=%d", srs_schema0, ret);
        return ret;
    }
    // try schema1
    bool is_valid = false;
    if ((ret = c1.c1_validate_digest(is_valid)) != ERROR_SUCCESS || !is_valid) {
1215
        srs_info("schema0 failed, try schema1.");
1216
        if ((ret = c1.parse(hs_bytes->c0c1 + 1, 1536, srs_schema1)) != ERROR_SUCCESS) {
1217 1218 1219 1220 1221 1222 1223 1224 1225
            srs_error("parse c1 schema%d error. ret=%d", srs_schema1, ret);
            return ret;
        }
        
        if ((ret = c1.c1_validate_digest(is_valid)) != ERROR_SUCCESS || !is_valid) {
            ret = ERROR_RTMP_TRY_SIMPLE_HS;
            srs_info("all schema valid failed, try simple handshake. ret=%d", ret);
            return ret;
        }
1226 1227
    } else {
        srs_info("schema0 is ok.");
1228 1229 1230 1231 1232 1233 1234 1235 1236 1237
    }
    srs_verbose("decode c1 success.");
    
    // encode s1
    c1s1 s1;
    if ((ret = s1.s1_create(&c1)) != ERROR_SUCCESS) {
        srs_error("create s1 from c1 failed. ret=%d", ret);
        return ret;
    }
    srs_verbose("create s1 from c1 success.");
1238 1239 1240
    // verify s1
    if ((ret = s1.s1_validate_digest(is_valid)) != ERROR_SUCCESS || !is_valid) {
        ret = ERROR_RTMP_TRY_SIMPLE_HS;
1241
        srs_info("verify s1 failed, try simple handshake. ret=%d", ret);
1242 1243
        return ret;
    }
1244
    srs_verbose("verify s1 success.");
1245 1246 1247 1248 1249 1250 1251
    
    c2s2 s2;
    if ((ret = s2.s2_create(&c1)) != ERROR_SUCCESS) {
        srs_error("create s2 from c1 failed. ret=%d", ret);
        return ret;
    }
    srs_verbose("create s2 from c1 success.");
1252 1253 1254 1255 1256 1257 1258
    // verify s2
    if ((ret = s2.s2_validate(&c1, is_valid)) != ERROR_SUCCESS || !is_valid) {
        ret = ERROR_RTMP_TRY_SIMPLE_HS;
        srs_info("verify s2 failed, try simple handshake. ret=%d", ret);
        return ret;
    }
    srs_verbose("verify s2 success.");
1259 1260
    
    // sendout s0s1s2
1261 1262 1263
    if ((ret = hs_bytes->create_s0s1s2()) != ERROR_SUCCESS) {
        return ret;
    }
1264 1265 1266 1267 1268 1269
    if ((ret = s1.dump(hs_bytes->s0s1s2 + 1, 1536)) != ERROR_SUCCESS) {
        return ret;
    }
    if ((ret = s2.dump(hs_bytes->s0s1s2 + 1537, 1536)) != ERROR_SUCCESS) {
        return ret;
    }
1270
    if ((ret = io->write(hs_bytes->s0s1s2, 3073, &nsize)) != ERROR_SUCCESS) {
winlin authored
1271 1272 1273 1274 1275 1276
        srs_warn("complex handshake send s0s1s2 failed. ret=%d", ret);
        return ret;
    }
    srs_verbose("complex handshake send s0s1s2 success.");
    
    // recv c2
1277
    if ((ret = hs_bytes->read_c2(io)) != ERROR_SUCCESS) {
winlin authored
1278 1279
        return ret;
    }
1280
    c2s2 c2;
1281 1282 1283
    if ((ret = c2.parse(hs_bytes->c2, 1536)) != ERROR_SUCCESS) {
        return ret;
    }
winlin authored
1284
    srs_verbose("complex handshake read c2 success.");
1285
    
1286
    // verify c2
1287 1288
    // never verify c2, for ffmpeg will failed.
    // it's ok for flash.
1289
    
winlin authored
1290
    srs_trace("complex handshake success");
1291 1292
    
    return ret;
winlin authored
1293
}
1294
#endif
winlin authored
1295
1296
#ifndef SRS_AUTO_SSL
1297
int SrsComplexHandshake::handshake_with_server(SrsHandshakeBytes* /*hs_bytes*/, ISrsProtocolReaderWriter* /*io*/)
winlin authored
1298
{
1299
    return ERROR_RTMP_TRY_SIMPLE_HS;
winlin authored
1300 1301
}
#else
1302
int SrsComplexHandshake::handshake_with_server(SrsHandshakeBytes* hs_bytes, ISrsProtocolReaderWriter* io)
winlin authored
1303
{
1304
    int ret = ERROR_SUCCESS;
1305
1306 1307 1308 1309 1310 1311 1312 1313 1314
    ssize_t nsize;
    
    // complex handshake
    if ((ret = hs_bytes->create_c0c1()) != ERROR_SUCCESS) {
        return ret;
    }
    
    // sign c1
    c1s1 c1;
1315 1316
    // @remark, FMS requires the schema1(digest-key), or connect failed.
    if ((ret = c1.c1_create(srs_schema1)) != ERROR_SUCCESS) {
1317 1318
        return ret;
    }
1319 1320 1321
    if ((ret = c1.dump(hs_bytes->c0c1 + 1, 1536)) != ERROR_SUCCESS) {
        return ret;
    }
1322
1323 1324 1325 1326 1327 1328 1329 1330 1331 1332 1333 1334 1335 1336 1337 1338 1339 1340 1341 1342 1343 1344 1345 1346 1347 1348 1349
    // verify c1
    bool is_valid;
    if ((ret = c1.c1_validate_digest(is_valid)) != ERROR_SUCCESS || !is_valid) {
        ret = ERROR_RTMP_TRY_SIMPLE_HS;
        return ret;
    }
    
    if ((ret = io->write(hs_bytes->c0c1, 1537, &nsize)) != ERROR_SUCCESS) {
        srs_warn("write c0c1 failed. ret=%d", ret);
        return ret;
    }
    srs_verbose("write c0c1 success.");
    
    // s0s1s2
    if ((ret = hs_bytes->read_s0s1s2(io)) != ERROR_SUCCESS) {
        return ret;
    }
    
    // plain text required.
    if (hs_bytes->s0s1s2[0] != 0x03) {
        ret = ERROR_RTMP_HANDSHAKE;
        srs_warn("handshake failed, plain text required. ret=%d", ret);
        return ret;
    }
    
    // verify s1s2
    c1s1 s1;
1350
    if ((ret = s1.parse(hs_bytes->s0s1s2 + 1, 1536, c1.schema())) != ERROR_SUCCESS) {
1351 1352
        return ret;
    }
1353 1354 1355 1356
    
    // never verify the s1,
    // for if forward to nginx-rtmp, verify s1 will failed,
    // TODO: FIXME: find the handshake schema of nginx-rtmp.
1357 1358 1359 1360 1361
    
    // c2
    if ((ret = hs_bytes->create_c2()) != ERROR_SUCCESS) {
        return ret;
    }
1362
1363 1364 1365 1366
    c2s2 c2;
    if ((ret = c2.c2_create(&s1)) != ERROR_SUCCESS) {
        return ret;
    }
1367
1368 1369 1370
    if ((ret = c2.dump(hs_bytes->c2, 1536)) != ERROR_SUCCESS) {
        return ret;
    }
1371 1372 1373 1374 1375 1376
    if ((ret = io->write(hs_bytes->c2, 1536, &nsize)) != ERROR_SUCCESS) {
        srs_warn("complex handshake write c2 failed. ret=%d", ret);
        return ret;
    }
    srs_verbose("complex handshake write c2 success.");
    
winlin authored
1377
    srs_trace("complex handshake success.");
1378 1379
    
    return ret;
winlin authored
1380 1381 1382
}
#endif
1383