Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(447)

Side by Side Diff: webrtc/pc/mediasession_unittest.cc

Issue 1956343002: Initial asymmetric codec support in MediaSessionDescription (Closed) Base URL: https://chromium.googlesource.com/external/webrtc.git@master
Patch Set: Check stream type to decide on whether or not to receive. Created 4 years, 7 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
« webrtc/pc/mediasession.cc ('K') | « webrtc/pc/mediasession.cc ('k') | no next file » | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
1 /* 1 /*
2 * Copyright 2004 The WebRTC project authors. All Rights Reserved. 2 * Copyright 2004 The WebRTC project authors. All Rights Reserved.
3 * 3 *
4 * Use of this source code is governed by a BSD-style license 4 * Use of this source code is governed by a BSD-style license
5 * that can be found in the LICENSE file in the root of the source 5 * that can be found in the LICENSE file in the root of the source
6 * tree. An additional intellectual property rights grant can be found 6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may 7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree. 8 * be found in the AUTHORS file in the root of the source tree.
9 */ 9 */
10 10
(...skipping 21 matching lines...) Expand all
32 #else 32 #else
33 #define ASSERT_CRYPTO(cd, s, cs) \ 33 #define ASSERT_CRYPTO(cd, s, cs) \
34 ASSERT_EQ(cricket::CT_NONE, cd->crypto_required()); \ 34 ASSERT_EQ(cricket::CT_NONE, cd->crypto_required()); \
35 ASSERT_EQ(0U, cd->cryptos().size()); 35 ASSERT_EQ(0U, cd->cryptos().size());
36 #endif 36 #endif
37 37
38 typedef std::vector<cricket::Candidate> Candidates; 38 typedef std::vector<cricket::Candidate> Candidates;
39 39
40 using cricket::MediaContentDescription; 40 using cricket::MediaContentDescription;
41 using cricket::MediaSessionDescriptionFactory; 41 using cricket::MediaSessionDescriptionFactory;
42 using cricket::MediaContentDirection;
42 using cricket::MediaSessionOptions; 43 using cricket::MediaSessionOptions;
43 using cricket::MediaType; 44 using cricket::MediaType;
44 using cricket::SessionDescription; 45 using cricket::SessionDescription;
45 using cricket::SsrcGroup; 46 using cricket::SsrcGroup;
46 using cricket::StreamParams; 47 using cricket::StreamParams;
47 using cricket::StreamParamsVec; 48 using cricket::StreamParamsVec;
48 using cricket::TransportDescription; 49 using cricket::TransportDescription;
49 using cricket::TransportDescriptionFactory; 50 using cricket::TransportDescriptionFactory;
50 using cricket::TransportInfo; 51 using cricket::TransportInfo;
51 using cricket::ContentInfo; 52 using cricket::ContentInfo;
(...skipping 2391 matching lines...) Expand 10 before | Expand all | Expand 10 after
2443 EXPECT_EQ(GetParam(), acd->protocol()); 2444 EXPECT_EQ(GetParam(), acd->protocol());
2444 EXPECT_EQ(GetParam(), vcd->protocol()); 2445 EXPECT_EQ(GetParam(), vcd->protocol());
2445 } 2446 }
2446 2447
2447 INSTANTIATE_TEST_CASE_P(MediaProtocolPatternTest, 2448 INSTANTIATE_TEST_CASE_P(MediaProtocolPatternTest,
2448 MediaProtocolTest, 2449 MediaProtocolTest,
2449 ::testing::ValuesIn(kMediaProtocols)); 2450 ::testing::ValuesIn(kMediaProtocols));
2450 INSTANTIATE_TEST_CASE_P(MediaProtocolDtlsPatternTest, 2451 INSTANTIATE_TEST_CASE_P(MediaProtocolDtlsPatternTest,
2451 MediaProtocolTest, 2452 MediaProtocolTest,
2452 ::testing::ValuesIn(kMediaProtocolsDtls)); 2453 ::testing::ValuesIn(kMediaProtocolsDtls));
2454
2455 TEST_F(MediaSessionDescriptionFactoryTest, TestSetAudioCodecsOld) {
2456 TransportDescriptionFactory tdf;
2457 MediaSessionDescriptionFactory sf(&tdf);
2458 const std::vector<AudioCodec> codecs = MAKE_VECTOR(kAudioCodecs1);
2459 sf.set_audio_codecs(codecs);
2460
2461 EXPECT_TRUE(sf.audio_codecs() == codecs);
2462 EXPECT_TRUE(sf.audio_send_codecs() == codecs);
2463 EXPECT_TRUE(sf.audio_recv_codecs() == codecs);
2464 }
2465
2466 TEST_F(MediaSessionDescriptionFactoryTest, TestSetAudioCodecs) {
2467 TransportDescriptionFactory tdf;
2468 MediaSessionDescriptionFactory sf(&tdf);
2469 std::vector<AudioCodec> send_codecs = MAKE_VECTOR(kAudioCodecs1);
2470 std::vector<AudioCodec> recv_codecs = MAKE_VECTOR(kAudioCodecs2);
2471
2472 // The merged list of codecs should contain any send codecs that are also
2473 // nominally in the recieve codecs list. Payload types should be picked from
2474 // the send codecs and a number-of-channels of 0 and 1 should be equivalent
2475 // (set to 1). This equals what happens when the send codecs are used in an
2476 // offer and the receive codecs are used in the following answer.
2477 const std::vector<AudioCodec> sendrecv_codecs =
2478 MAKE_VECTOR(kAudioCodecsAnswer);
2479 const std::vector<AudioCodec> no_codecs;
2480
2481 RTC_CHECK_EQ(send_codecs[1].name, "iLBC")
2482 << "Please don't change shared test data!";
2483 RTC_CHECK_EQ(recv_codecs[2].name, "iLBC")
2484 << "Please don't change shared test data!";
2485 // Alter iLBC send codec to have zero channels, to test that that is handled
2486 // properly.
2487 send_codecs[1].channels = 0;
2488
2489 // Alther iLBC receive codec to be lowercase, to test that case conversions
2490 // are handled properly.
2491 recv_codecs[2].name = "ilbc";
2492
2493 // Test proper merge
2494 sf.set_audio_codecs(send_codecs, recv_codecs);
2495 EXPECT_TRUE(sf.audio_send_codecs() == send_codecs);
2496 EXPECT_TRUE(sf.audio_recv_codecs() == recv_codecs);
2497 EXPECT_TRUE(sf.audio_codecs() == sendrecv_codecs);
2498
2499 // Test empty send codecs list
2500 sf.set_audio_codecs(no_codecs, recv_codecs);
2501 EXPECT_TRUE(sf.audio_send_codecs() == no_codecs);
2502 EXPECT_TRUE(sf.audio_recv_codecs() == recv_codecs);
2503 EXPECT_TRUE(sf.audio_codecs() == no_codecs);
2504
2505 // Test empty recv codecs list
2506 sf.set_audio_codecs(send_codecs, no_codecs);
2507 EXPECT_TRUE(sf.audio_send_codecs() == send_codecs);
2508 EXPECT_TRUE(sf.audio_recv_codecs() == no_codecs);
2509 EXPECT_TRUE(sf.audio_codecs() == no_codecs);
2510
2511 // Test all empty codec lists
2512 sf.set_audio_codecs(no_codecs, no_codecs);
2513 EXPECT_TRUE(sf.audio_send_codecs() == no_codecs);
2514 EXPECT_TRUE(sf.audio_recv_codecs() == no_codecs);
2515 EXPECT_TRUE(sf.audio_codecs() == no_codecs);
2516 }
2517
2518 namespace {
2519 void TestAudioCodecsOffer(MediaContentDirection direction,
2520 bool add_legacy_stream) {
2521 TransportDescriptionFactory tdf;
2522 MediaSessionDescriptionFactory sf(&tdf);
2523 const std::vector<AudioCodec> send_codecs = MAKE_VECTOR(kAudioCodecs1);
2524 const std::vector<AudioCodec> recv_codecs = MAKE_VECTOR(kAudioCodecs2);
2525 const std::vector<AudioCodec> sendrecv_codecs =
2526 MAKE_VECTOR(kAudioCodecsAnswer);
2527 sf.set_audio_codecs(send_codecs, recv_codecs);
2528 sf.set_add_legacy_streams(add_legacy_stream);
2529
2530 MediaSessionOptions opts;
2531 opts.recv_audio = (direction == cricket::MD_RECVONLY ||
2532 direction == cricket::MD_SENDRECV);
2533 opts.recv_video = false;
2534 if (direction == cricket::MD_SENDONLY || direction == cricket::MD_SENDRECV)
2535 opts.AddSendStream(MEDIA_TYPE_AUDIO, kAudioTrack1, kMediaStream1);
2536
2537 std::unique_ptr<SessionDescription> offer(sf.CreateOffer(opts, NULL));
2538 ASSERT_TRUE(offer.get() != NULL);
2539 const ContentInfo* ac = offer->GetContentByName("audio");
2540
2541 // If the factory didn't add any audio content to the offer, we cannot check
2542 // that the codecs put in are right. This happens when we neither want to send
2543 // nor receive audio. The checks are still in place if at some point we'd
2544 // instead create an inactive stream.
2545 if (ac) {
2546 AudioContentDescription* acd =
2547 static_cast<AudioContentDescription*>(ac->description);
2548 // sendrecv and inactive should both present lists as if the channel was to
2549 // be used for sending and receiving. Inactive essentially means it might
2550 // eventually be used anything, but we don't know more at this moment.
2551 if (acd->direction() == cricket::MD_SENDONLY) {
2552 EXPECT_TRUE(acd->codecs() == send_codecs);
2553 } else if (acd->direction() == cricket::MD_RECVONLY) {
2554 EXPECT_TRUE(acd->codecs() == recv_codecs);
2555 } else {
2556 EXPECT_TRUE(acd->codecs() == sendrecv_codecs);
2557 }
2558 }
2559 }
2560
2561 static const AudioCodec kOfferAnswerCodecs[] = {
2562 AudioCodec(0, "codec0", 16000, -1, 1),
2563 AudioCodec(1, "codec1", 8000, 13300, 1),
2564 AudioCodec(2, "codec2", 8000, 64000, 1),
2565 AudioCodec(3, "codec3", 8000, 64000, 1),
2566 AudioCodec(4, "codec4", 8000, 0, 2),
2567 AudioCodec(5, "codec5", 32000, 0, 1),
2568 AudioCodec(6, "codec6", 48000, 0, 1)
2569 };
2570
2571 // Codecs used by offerer in the AudioCodecsAnswerTest
2572 static const int kOfferSendCodecs[] = { 0, 1, 3, 5, 6 };
2573 static const int kOfferRecvCodecs[] = { 1, 2, 3, 4, 6 };
2574 // Codecs used in the answerer in the AudioCodecsAnswerTest. The order is
2575 // jumbled to catch the answer not following the order in the offer.
2576 static const int kAnswerSendCodecs[] = { 6, 5, 2, 3, 4 };
2577 static const int kAnswerRecvCodecs[] = { 6, 5, 4, 1, 0 };
2578 // The resulting sets of codecs in the answer in the AudioCodecsAnswerTest
2579 static const int kResultSend_RecvCodecs[] = { 0, 1, 5, 6 };
2580 static const int kResultSend_SendrecvCodecs[] = { 5, 6 };
2581 static const int kResultRecv_SendCodecs[] = { 2, 3, 4, 6 };
2582 static const int kResultRecv_SendrecvCodecs[] = { 4, 6 };
2583 static const int kResultSendrecv_SendCodecs[] = { 3, 6 };
2584 static const int kResultSendrecv_RecvCodecs[] = { 1, 6 };
2585 static const int kResultSendrecv_SendrecvCodecs[] = { 6 };
2586
2587 template <typename T, int IDXS>
2588 std::vector<T> VectorFromIndices(const T* array, const int (&indices)[IDXS]) {
2589 std::vector<T> out;
2590 out.reserve(IDXS);
2591 for (int idx : indices)
2592 out.push_back(array[idx]);
2593
2594 return out;
2595 }
2596
2597 void TestAudioCodecsAnswer(MediaContentDirection offer_direction,
2598 MediaContentDirection answer_direction,
2599 bool add_legacy_stream) {
2600 TransportDescriptionFactory offer_tdf;
2601 TransportDescriptionFactory answer_tdf;
2602 MediaSessionDescriptionFactory offer_factory(&offer_tdf);
2603 MediaSessionDescriptionFactory answer_factory(&answer_tdf);
2604 offer_factory.set_audio_codecs(
2605 VectorFromIndices(kOfferAnswerCodecs, kOfferSendCodecs),
2606 VectorFromIndices(kOfferAnswerCodecs, kOfferRecvCodecs));
2607 answer_factory.set_audio_codecs(
2608 VectorFromIndices(kOfferAnswerCodecs, kAnswerSendCodecs),
2609 VectorFromIndices(kOfferAnswerCodecs, kAnswerRecvCodecs));
2610
2611 // Never add a legacy stream to offer - we want to control the offer
2612 // parameters exactly.
2613 offer_factory.set_add_legacy_streams(false);
2614 answer_factory.set_add_legacy_streams(add_legacy_stream);
2615 MediaSessionOptions offer_opts;
2616 offer_opts.recv_audio = (offer_direction == cricket::MD_RECVONLY ||
2617 offer_direction == cricket::MD_SENDRECV);
2618 offer_opts.recv_video = false;
2619 if (offer_direction == cricket::MD_SENDONLY ||
2620 offer_direction == cricket::MD_SENDRECV) {
2621 offer_opts.AddSendStream(MEDIA_TYPE_AUDIO, kAudioTrack1, kMediaStream1);
2622 }
2623
2624 std::unique_ptr<SessionDescription> offer(
2625 offer_factory.CreateOffer(offer_opts, NULL));
2626 ASSERT_TRUE(offer.get() != NULL);
2627
2628 MediaSessionOptions answer_opts;
2629 answer_opts.recv_audio = (answer_direction == cricket::MD_RECVONLY ||
2630 answer_direction == cricket::MD_SENDRECV);
2631 answer_opts.recv_video = false;
2632 if (answer_direction == cricket::MD_SENDONLY ||
2633 answer_direction == cricket::MD_SENDRECV) {
2634 answer_opts.AddSendStream(MEDIA_TYPE_AUDIO, kAudioTrack1, kMediaStream1);
2635 }
2636 std::unique_ptr<SessionDescription> answer(
2637 answer_factory.CreateAnswer(offer.get(), answer_opts, NULL));
2638 const ContentInfo* ac = answer->GetContentByName("audio");
2639
2640 // If the factory didn't add any audio content to the answer, we cannot check
2641 // that the codecs put in are right. This happens when we neither want to send
2642 // nor receive audio. The checks are still in place if at some point we'd
2643 // instead create an inactive stream.
2644 if (ac) {
2645 const AudioContentDescription* acd =
2646 static_cast<const AudioContentDescription*>(ac->description);
2647 EXPECT_EQ(MEDIA_TYPE_AUDIO, acd->type());
2648
2649
2650 std::vector<AudioCodec> target_codecs;
2651 // For offers with sendrecv or inactive, we should never reply with more
2652 // codecs than offered, with these codec sets.
2653 switch (offer_direction) {
2654 case cricket::MD_INACTIVE:
2655 target_codecs = VectorFromIndices(kOfferAnswerCodecs,
2656 kResultSendrecv_SendrecvCodecs);
2657 break;
2658 case cricket::MD_SENDONLY:
2659 if (acd->direction() == cricket::MD_RECVONLY) {
2660 target_codecs = VectorFromIndices(kOfferAnswerCodecs,
2661 kResultSend_RecvCodecs);
2662 } else {
2663 target_codecs = VectorFromIndices(kOfferAnswerCodecs,
2664 kResultSend_SendrecvCodecs);
2665 }
2666 break;
2667 case cricket::MD_RECVONLY:
2668 if (acd->direction() == cricket::MD_SENDONLY) {
2669 target_codecs = VectorFromIndices(kOfferAnswerCodecs,
2670 kResultRecv_SendCodecs);
2671 } else {
2672 target_codecs = VectorFromIndices(kOfferAnswerCodecs,
2673 kResultRecv_SendrecvCodecs);
2674 }
2675 break;
2676 case cricket::MD_SENDRECV:
2677 if (acd->direction() == cricket::MD_SENDONLY) {
2678 target_codecs = VectorFromIndices(kOfferAnswerCodecs,
2679 kResultSendrecv_SendCodecs);
2680 } else if (acd->direction() == cricket::MD_RECVONLY) {
2681 target_codecs = VectorFromIndices(kOfferAnswerCodecs,
2682 kResultSendrecv_RecvCodecs);
2683 } else {
2684 target_codecs = VectorFromIndices(kOfferAnswerCodecs,
2685 kResultSendrecv_SendrecvCodecs);
2686 }
2687 break;
2688 }
2689
2690 auto format_codecs = [] (const std::vector<AudioCodec>& codecs) {
2691 std::stringstream os;
2692 bool first = true;
2693 os << "{";
2694 for (const auto& c : codecs) {
2695 os << (first ? " " : ", ") << c.id;
2696 first = false;
2697 }
2698 os << " }";
2699 return os.str();
2700 };
2701
2702 EXPECT_TRUE(acd->codecs() == target_codecs)
2703 << "Expected: " << format_codecs(target_codecs)
2704 << ", got: " << format_codecs(acd->codecs())
2705 << "; Offered: " << MediaContentDirectionToString(offer_direction)
2706 << ", answerer wants: "
2707 << MediaContentDirectionToString(answer_direction)
2708 << "; got: " << MediaContentDirectionToString(acd->direction());
2709 } else {
2710 EXPECT_EQ(offer_direction, cricket::MD_INACTIVE)
2711 << "Only inactive offers are allowed to not generate any audio content";
2712 }
2713 }
2714 }
2715
2716 class AudioCodecsOfferTest
2717 : public ::testing::TestWithParam<std::tr1::tuple<MediaContentDirection,
2718 bool>> {
2719 };
2720
2721 TEST_P(AudioCodecsOfferTest, TestCodecsInOffer) {
2722 TestAudioCodecsOffer(std::tr1::get<0>(GetParam()),
2723 std::tr1::get<1>(GetParam()));
2724 }
2725
2726 INSTANTIATE_TEST_CASE_P(MediaSessionDescriptionFactoryTest,
2727 AudioCodecsOfferTest,
2728 ::testing::Combine(
2729 ::testing::Values(cricket::MD_SENDONLY,
2730 cricket::MD_RECVONLY,
2731 cricket::MD_SENDRECV,
2732 cricket::MD_INACTIVE),
2733 ::testing::Bool()));
2734
2735 class AudioCodecsAnswerTest
2736 : public ::testing::TestWithParam<std::tr1::tuple<MediaContentDirection,
2737 MediaContentDirection,
2738 bool>> {
2739 };
2740
2741 TEST_P(AudioCodecsAnswerTest, TestCodecsInAnswer) {
2742 TestAudioCodecsAnswer(std::tr1::get<0>(GetParam()),
2743 std::tr1::get<1>(GetParam()),
2744 std::tr1::get<2>(GetParam()));
2745 }
2746
2747 INSTANTIATE_TEST_CASE_P(MediaSessionDescriptionFactoryTest,
2748 AudioCodecsAnswerTest,
2749 ::testing::Combine(
2750 ::testing::Values(cricket::MD_SENDONLY,
2751 cricket::MD_RECVONLY,
2752 cricket::MD_SENDRECV,
2753 cricket::MD_INACTIVE),
2754 ::testing::Values(cricket::MD_SENDONLY,
2755 cricket::MD_RECVONLY,
2756 cricket::MD_SENDRECV,
2757 cricket::MD_INACTIVE),
2758 ::testing::Bool()));
OLDNEW
« webrtc/pc/mediasession.cc ('K') | « webrtc/pc/mediasession.cc ('k') | no next file » | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698