<p>Jenkins2 <strong>merged</strong> this change.</p><p><a href="https://gerrit.asterisk.org/8782">View Change</a></p><div style="white-space:pre-wrap">Approvals:
  Richard Mudgett: Looks good to me, but someone else must approve
  George Joseph: Looks good to me, approved
  Jenkins2: Approved for Submit

</div><pre style="font-family: monospace,monospace; white-space: pre-wrap;">bridge_softmix / app_confbridge: Add support for REMB combining.<br><br>This change adds the ability for multiple REMB reports in<br>bridge_softmix to be combined according to a configured<br>behavior into a single report. This single report is sent<br>back to the sender of video, which adjusts the encoding bitrate<br>to be at or below the bitrate of the report. The available<br>behaviors are: lowest, highest, and average. Lowest uses the<br>lowest received bitrate. Highest uses the highest received<br>bitrate. Average goes through the received bitrates adding<br>them to the previous average and creates a new average.<br><br>Other behaviors can be added in the future and the existing<br>average one may be adjusted, but this provides the foundation<br>to do so.<br><br>Support for configuring which behavior to use has been<br>added to app_confbridge.<br><br>ASTERISK-27804<br><br>Change-Id: I9eafe4e7c1f72d67074a8d6acb26bfcf19322b66<br>---<br>M apps/app_confbridge.c<br>M apps/confbridge/conf_config_parser.c<br>M apps/confbridge/include/confbridge.h<br>M bridges/bridge_softmix.c<br>M bridges/bridge_softmix/include/bridge_softmix_internal.h<br>M configs/samples/confbridge.conf.sample<br>M include/asterisk/bridge.h<br>M main/bridge.c<br>8 files changed, 376 insertions(+), 3 deletions(-)<br><br></pre><pre style="font-family: monospace,monospace; white-space: pre-wrap;">diff --git a/apps/app_confbridge.c b/apps/app_confbridge.c<br>index 5907bf9..f3f0fdf 100644<br>--- a/apps/app_confbridge.c<br>+++ b/apps/app_confbridge.c<br>@@ -1544,6 +1544,13 @@<br>                     ast_bridge_set_sfu_video_mode(conference->bridge);<br>                         ast_bridge_set_video_update_discard(conference->bridge, conference->b_profile.video_update_discard);<br>                    ast_bridge_set_remb_send_interval(conference->bridge, conference->b_profile.remb_send_interval);<br>+                       if (ast_test_flag(&conference->b_profile, BRIDGE_OPT_REMB_BEHAVIOR_AVERAGE)) {<br>+                                ast_brige_set_remb_behavior(conference->bridge, AST_BRIDGE_VIDEO_SFU_REMB_AVERAGE);<br>+                       } else if (ast_test_flag(&conference->b_profile, BRIDGE_OPT_REMB_BEHAVIOR_LOWEST)) {<br>+                          ast_brige_set_remb_behavior(conference->bridge, AST_BRIDGE_VIDEO_SFU_REMB_LOWEST);<br>+                        } else if (ast_test_flag(&conference->b_profile, BRIDGE_OPT_REMB_BEHAVIOR_HIGHEST)) {<br>+                         ast_brige_set_remb_behavior(conference->bridge, AST_BRIDGE_VIDEO_SFU_REMB_HIGHEST);<br>+                       }<br>             }<br> <br>          /* Link it into the conference bridges container */<br>diff --git a/apps/confbridge/conf_config_parser.c b/apps/confbridge/conf_config_parser.c<br>index f9d7483..c143e39 100644<br>--- a/apps/confbridge/conf_config_parser.c<br>+++ b/apps/confbridge/conf_config_parser.c<br>@@ -470,6 +470,27 @@<br>                                            better quality for all receivers.<br>                                     </para></description><br>                             </configOption><br>+                                <configOption name="remb_behavior" default="average"><br>+                                      <synopsis>Sets how REMB reports are generated from multiple sources</synopsis><br>+                                   <description><para><br>+                                              Sets how REMB reports are combined from multiple sources to form one. A REMB report<br>+                                          consists of information about the receiver estimated maximum bitrate. As a source<br>+                                            stream may be forwarded to multiple receivers the reports must be combined into<br>+                                              a single one which is sent to the sender.</para><br>+                                               <enumlist><br>+                                                     <enum name="average"><br>+                                                                <para>The average of all estimated maximum bitrates is taken and sent<br>+                                                          to the sender.</para><br>+                                                  </enum><br>+                                                        <enum name="lowest"><br>+                                                         <para>The lowest estimated maximum bitrate is forwarded to the sender.</para><br>+                                                    </enum><br>+                                                        <enum name="highest"><br>+                                                                <para>The highest estimated maximum bitrate is forwarded to the sender.</para><br>+                                                   </enum><br>+                                                </enumlist><br>+                                    </description><br>+                         </configOption><br>                                 <configOption name="template"><br>                                        <synopsis>When using the CONFBRIDGE dialplan function, use a bridge profile as a template for creating a new temporary profile</synopsis><br>                                 </configOption><br>@@ -1675,6 +1696,23 @@<br>         ast_cli(a->fd,"Video Update Discard: %u\n", b_profile.video_update_discard);<br>     ast_cli(a->fd,"REMB Send Interval: %u\n", b_profile.remb_send_interval);<br> <br>+     switch (b_profile.flags<br>+              & (BRIDGE_OPT_REMB_BEHAVIOR_AVERAGE | BRIDGE_OPT_REMB_BEHAVIOR_LOWEST<br>+                    | BRIDGE_OPT_REMB_BEHAVIOR_HIGHEST)) {<br>+       case BRIDGE_OPT_REMB_BEHAVIOR_AVERAGE:<br>+               ast_cli(a->fd, "REMB Behavior:           average\n");<br>+           break;<br>+       case BRIDGE_OPT_REMB_BEHAVIOR_LOWEST:<br>+                ast_cli(a->fd, "REMB Behavior:           lowest\n");<br>+            break;<br>+       case BRIDGE_OPT_REMB_BEHAVIOR_HIGHEST:<br>+               ast_cli(a->fd, "REMB Behavior:           highest\n");<br>+           break;<br>+       default:<br>+             ast_assert(0);<br>+               break;<br>+       }<br>+<br>  ast_cli(a->fd,"sound_only_person:    %s\n", conf_get_sound(CONF_SOUND_ONLY_PERSON, b_profile.sounds));<br>   ast_cli(a->fd,"sound_only_one:       %s\n", conf_get_sound(CONF_SOUND_ONLY_ONE, b_profile.sounds));<br>      ast_cli(a->fd,"sound_has_joined:     %s\n", conf_get_sound(CONF_SOUND_HAS_JOINED, b_profile.sounds));<br>@@ -2011,6 +2049,30 @@<br>    return 0;<br> }<br> <br>+static int remb_behavior_handler(const struct aco_option *opt, struct ast_variable *var, void *obj)<br>+{<br>+   struct bridge_profile *b_profile = obj;<br>+<br>+   if (strcasecmp(var->name, "remb_behavior")) {<br>+           return -1;<br>+   }<br>+<br>+ ast_clear_flag(b_profile, BRIDGE_OPT_REMB_BEHAVIOR_AVERAGE |<br>+         BRIDGE_OPT_REMB_BEHAVIOR_LOWEST |<br>+            BRIDGE_OPT_REMB_BEHAVIOR_HIGHEST);<br>+<br>+        if (!strcasecmp(var->value, "average")) {<br>+               ast_set_flag(b_profile, BRIDGE_OPT_REMB_BEHAVIOR_AVERAGE);<br>+   } else if (!strcasecmp(var->value, "lowest")) {<br>+         ast_set_flag(b_profile, BRIDGE_OPT_REMB_BEHAVIOR_LOWEST);<br>+    } else if (!strcasecmp(var->value, "highest")) {<br>+                ast_set_flag(b_profile, BRIDGE_OPT_REMB_BEHAVIOR_HIGHEST);<br>+   } else {<br>+             return -1;<br>+   }<br>+    return 0;<br>+}<br>+<br> static int user_template_handler(const struct aco_option *opt, struct ast_variable *var, void *obj)<br> {<br>    struct user_profile *u_profile = obj;<br>@@ -2245,6 +2307,7 @@<br>  aco_option_register_custom(&cfg_info, "sound_", ACO_PREFIX, bridge_types, NULL, sound_option_handler, 0);<br>       aco_option_register(&cfg_info, "video_update_discard", ACO_EXACT, bridge_types, "2000", OPT_UINT_T, 0, FLDSET(struct bridge_profile, video_update_discard));<br>  aco_option_register(&cfg_info, "remb_send_interval", ACO_EXACT, bridge_types, "0", OPT_UINT_T, 0, FLDSET(struct bridge_profile, remb_send_interval));<br>+        aco_option_register_custom(&cfg_info, "remb_behavior", ACO_EXACT, bridge_types, "average", remb_behavior_handler, 0);<br>         /* This option should only be used with the CONFBRIDGE dialplan function */<br>   aco_option_register_custom(&cfg_info, "template", ACO_EXACT, bridge_types, NULL, bridge_template_handler, 0);<br> <br>diff --git a/apps/confbridge/include/confbridge.h b/apps/confbridge/include/confbridge.h<br>index c2f8f9a..0a0a571 100644<br>--- a/apps/confbridge/include/confbridge.h<br>+++ b/apps/confbridge/include/confbridge.h<br>@@ -76,6 +76,9 @@<br>    BRIDGE_OPT_RECORD_FILE_TIMESTAMP = (1 << 5), /*!< Set if the record file should have a timestamp appended */<br>         BRIDGE_OPT_BINAURAL_ACTIVE = (1 << 6), /*!< Set if binaural convolution is activated */<br>      BRIDGE_OPT_VIDEO_SRC_SFU = (1 << 7), /*!< Selective forwarding unit */<br>+      BRIDGE_OPT_REMB_BEHAVIOR_AVERAGE = (1 << 8), /*!< The average of all REMB reports is sent to the sender */<br>+  BRIDGE_OPT_REMB_BEHAVIOR_LOWEST = (1 << 9), /*!< The lowest estimated maximum bitrate is sent to the sender */<br>+      BRIDGE_OPT_REMB_BEHAVIOR_HIGHEST = (1 << 10), /*!< The highest estimated maximum bitrate is sent to the sender */<br> };<br> <br> enum conf_menu_action_id {<br>diff --git a/bridges/bridge_softmix.c b/bridges/bridge_softmix.c<br>index 16e1fb8..f0a3fb4 100644<br>--- a/bridges/bridge_softmix.c<br>+++ b/bridges/bridge_softmix.c<br>@@ -69,6 +69,15 @@<br> #define SOFTBRIDGE_VIDEO_DEST_LEN strlen(SOFTBRIDGE_VIDEO_DEST_PREFIX)<br> #define SOFTBRIDGE_VIDEO_DEST_SEPARATOR '_'<br> <br>+struct softmix_remb_collector {<br>+     /*! The frame which will be given to each source stream */<br>+   struct ast_frame frame;<br>+      /*! The REMB to send to the source which is collecting REMB reports */<br>+       struct ast_rtp_rtcp_feedback feedback;<br>+       /*! The maximum bitrate */<br>+   unsigned int bitrate;<br>+};<br>+<br> struct softmix_stats {<br>        /*! Each index represents a sample rate used above the internal rate. */<br>      unsigned int sample_rates[16];<br>@@ -768,6 +777,10 @@<br> <br>       ast_stream_topology_free(sc->topology);<br> <br>+        ao2_cleanup(sc->remb_collector);<br>+<br>+       AST_VECTOR_FREE(&sc->video_sources);<br>+<br>        /* Drop mutex lock */<br>         ast_mutex_destroy(&sc->lock);<br> <br>@@ -1160,6 +1173,39 @@<br> <br> /*!<br>  * \internal<br>+ * \brief Determine what to do with an RTCP frame.<br>+ * \since 15.4.0<br>+ *<br>+ * \param bridge Which bridge is getting the frame<br>+ * \param bridge_channel Which channel is writing the frame.<br>+ * \param frame What is being written.<br>+ */<br>+static void softmix_bridge_write_rtcp(struct ast_bridge *bridge, struct ast_bridge_channel *bridge_channel, struct ast_frame *frame)<br>+{<br>+   struct ast_rtp_rtcp_feedback *feedback = frame->data.ptr;<br>+ struct softmix_channel *sc = bridge_channel->tech_pvt;<br>+<br>+ /* We only care about REMB reports right now. In the future we may be able to use sender or<br>+   * receiver reports to further tweak things, but not yet.<br>+     */<br>+  if (frame->subclass.integer != AST_RTP_RTCP_PSFB || feedback->fmt != AST_RTP_RTCP_FMT_REMB ||<br>+          bridge->softmix.video_mode.mode != AST_BRIDGE_VIDEO_MODE_SFU ||<br>+           !bridge->softmix.video_mode.mode_data.sfu_data.remb_send_interval) {<br>+              return;<br>+      }<br>+<br>+ /* REMB is the total estimated maximum bitrate across all streams within the session, so we store<br>+     * only the latest report and use it everywhere.<br>+      */<br>+  ast_mutex_lock(&sc->lock);<br>+    sc->remb = feedback->remb;<br>+     ast_mutex_unlock(&sc->lock);<br>+<br>+       return;<br>+}<br>+<br>+/*!<br>+ * \internal<br>  * \brief Determine what to do with a frame written into the bridge.<br>  * \since 12.0.0<br>  *<br>@@ -1204,6 +1250,9 @@<br>     case AST_FRAME_CONTROL:<br>               res = softmix_bridge_write_control(bridge, bridge_channel, frame);<br>            break;<br>+       case AST_FRAME_RTCP:<br>+         softmix_bridge_write_rtcp(bridge, bridge_channel, frame);<br>+            break;<br>        case AST_FRAME_BRIDGE_ACTION:<br>                 res = ast_bridge_queue_everyone_else(bridge, bridge_channel, frame);<br>          break;<br>@@ -1217,6 +1266,104 @@<br>       }<br> <br>  return res;<br>+}<br>+<br>+static void remb_collect_report(struct ast_bridge *bridge, struct ast_bridge_channel *bridge_channel,<br>+   struct softmix_bridge_data *softmix_data, struct softmix_channel *sc)<br>+{<br>+    int i;<br>+       unsigned int bitrate;<br>+<br>+     /* If there are no video sources that we are a receiver of then we have noone to<br>+      * report REMB to.<br>+    */<br>+  if (!AST_VECTOR_SIZE(&sc->video_sources)) {<br>+           return;<br>+      }<br>+<br>+ /* We evenly divide the available maximum bitrate across the video sources<br>+    * to this receiver so each source gets an equal slice.<br>+       */<br>+  bitrate = (sc->remb.br_mantissa << sc->remb.br_exp) / AST_VECTOR_SIZE(&sc->video_sources);<br>+<br>+     /* If this receiver has no bitrate yet ignore it */<br>+  if (!bitrate) {<br>+              return;<br>+      }<br>+<br>+ for (i = 0; i < AST_VECTOR_SIZE(&sc->video_sources); ++i) {<br>+                struct softmix_remb_collector *collector;<br>+<br>+         /* The collector will always exist if a video source is in our list */<br>+               collector = AST_VECTOR_GET(&softmix_data->remb_collectors, AST_VECTOR_GET(&sc->video_sources, i));<br>+<br>+          if (!collector->bitrate) {<br>+                        collector->bitrate = bitrate;<br>+                     continue;<br>+            }<br>+<br>+         switch (bridge->softmix.video_mode.mode_data.sfu_data.remb_behavior) {<br>+            case AST_BRIDGE_VIDEO_SFU_REMB_AVERAGE:<br>+                      collector->bitrate = (collector->bitrate + bitrate) / 2;<br>+                       break;<br>+               case AST_BRIDGE_VIDEO_SFU_REMB_LOWEST:<br>+                       if (bitrate < collector->bitrate) {<br>+                            collector->bitrate = bitrate;<br>+                     }<br>+                    break;<br>+               case AST_BRIDGE_VIDEO_SFU_REMB_HIGHEST:<br>+                      if (bitrate > collector->bitrate) {<br>+                            collector->bitrate = bitrate;<br>+                     }<br>+                    break;<br>+               }<br>+    }<br>+}<br>+<br>+static void remb_send_report(struct ast_bridge_channel *bridge_channel, struct softmix_channel *sc)<br>+{<br>+   int i;<br>+<br>+    if (!sc->remb_collector) {<br>+                return;<br>+      }<br>+<br>+ /* If we have a new bitrate then use it for the REMB, if not we use the previous<br>+      * one until we know otherwise. This way the bitrate doesn't drop to 0 all of a sudden.<br>+   */<br>+  if (sc->remb_collector->bitrate) {<br>+             sc->remb_collector->feedback.remb.br_mantissa = sc->remb_collector->bitrate;<br>+             sc->remb_collector->feedback.remb.br_exp = 0;<br>+<br>+               /* The mantissa only has 18 bits available, so while it exceeds them we bump<br>+          * up the exp.<br>+                */<br>+          while (sc->remb_collector->feedback.remb.br_mantissa > 0x3ffff) {<br>+                   sc->remb_collector->feedback.remb.br_mantissa = sc->remb_collector->feedback.remb.br_mantissa >> 1;<br>+                        sc->remb_collector->feedback.remb.br_exp++;<br>+            }<br>+    }<br>+<br>+ for (i = 0; i < AST_VECTOR_SIZE(&bridge_channel->stream_map.to_bridge); ++i) {<br>+             int bridge_num = AST_VECTOR_GET(&bridge_channel->stream_map.to_bridge, i);<br>+<br>+         /* If this stream is not being provided to the bridge there can be no receivers of it<br>+                 * so therefore no REMB reports.<br>+              */<br>+          if (bridge_num == -1) {<br>+                      continue;<br>+            }<br>+<br>+         /* We need to update the frame with this stream, or else it won't be<br>+              * properly routed. We don't use the actual channel stream identifier as<br>+          * the bridging core will do the translation from bridge stream identifier to<br>+                 * channel stream identifier.<br>+                 */<br>+          sc->remb_collector->frame.stream_num = bridge_num;<br>+             ast_bridge_channel_queue_frame(bridge_channel, &sc->remb_collector->frame);<br>+        }<br>+<br>+ sc->remb_collector->bitrate = 0;<br> }<br> <br> static void gather_softmix_stats(struct softmix_stats *stats,<br>@@ -1440,6 +1587,7 @@<br>          struct ast_format *cur_slin = ast_format_cache_get_slin_by_rate(softmix_data->internal_rate);<br>              unsigned int softmix_samples = SOFTMIX_SAMPLES(softmix_data->internal_rate, softmix_data->internal_mixing_interval);<br>            unsigned int softmix_datalen = SOFTMIX_DATALEN(softmix_data->internal_rate, softmix_data->internal_mixing_interval);<br>+           int remb_update = 0;<br> <br>               if (softmix_datalen > MAX_DATALEN) {<br>                       /* This should NEVER happen, but if it does we need to know about it. Almost<br>@@ -1478,6 +1626,14 @@<br>          check_binaural_position_change(bridge, softmix_data);<br> #endif<br> <br>+            /* If we need to do a REMB update to all video sources then do so */<br>+         if (bridge->softmix.video_mode.mode == AST_BRIDGE_VIDEO_MODE_SFU &&<br>+                       bridge->softmix.video_mode.mode_data.sfu_data.remb_send_interval &&<br>+                       ast_tvdiff_ms(ast_tvnow(), softmix_data->last_remb_update) > bridge->softmix.video_mode.mode_data.sfu_data.remb_send_interval) {<br>+                    remb_update = 1;<br>+                     softmix_data->last_remb_update = ast_tvnow();<br>+             }<br>+<br>          /* Go through pulling audio from each factory that has it available */<br>                AST_LIST_TRAVERSE(&bridge->channels, bridge_channel, entry) {<br>                  struct softmix_channel *sc = bridge_channel->tech_pvt;<br>@@ -1511,6 +1667,9 @@<br>                                              ast_channel_name(bridge_channel->chan));<br> #endif<br>                          mixing_array.used_entries++;<br>+                 }<br>+                    if (remb_update) {<br>+                           remb_collect_report(bridge, bridge_channel, softmix_data, sc);<br>                        }<br>                     ast_mutex_unlock(&sc->lock);<br>           }<br>@@ -1562,6 +1721,10 @@<br> <br>                  /* A frame is now ready for the channel. */<br>                   ast_bridge_channel_queue_frame(bridge_channel, &sc->write_frame);<br>+<br>+                  if (remb_update) {<br>+                           remb_send_report(bridge_channel, sc);<br>+                        }<br>             }<br> <br>          update_all_rates = 0;<br>@@ -1688,6 +1851,8 @@<br>  }<br>     ast_mutex_destroy(&softmix_data->lock);<br>        ast_cond_destroy(&softmix_data->cond);<br>+        AST_VECTOR_RESET(&softmix_data->remb_collectors, ao2_cleanup);<br>+        AST_VECTOR_FREE(&softmix_data->remb_collectors);<br>       ast_free(softmix_data);<br> }<br> <br>@@ -1717,6 +1882,8 @@<br>         softmix_data->default_sample_size = SOFTMIX_SAMPLES(softmix_data->internal_rate,<br>                        softmix_data->internal_mixing_interval);<br> #endif<br>+<br>+      AST_VECTOR_INIT(&softmix_data->remb_collectors, 0);<br> <br>         bridge->tech_pvt = softmix_data;<br> <br>@@ -1814,12 +1981,67 @@<br> <br>                    stream = ast_stream_topology_get_stream(topology, i);<br>                         if (is_video_dest(stream, source_channel_name, source_stream_name)) {<br>+                                struct softmix_channel *sc = participant->tech_pvt;<br>+<br>                             AST_VECTOR_REPLACE(&participant->stream_map.to_channel, bridge_stream_position, i);<br>+                           AST_VECTOR_APPEND(&sc->video_sources, bridge_stream_position);<br>                                 break;<br>                        }<br>             }<br>             ast_channel_unlock(participant->chan);<br>             ast_bridge_channel_unlock(participant);<br>+      }<br>+}<br>+<br>+/*!<br>+ * \brief Allocate a REMB collector<br>+ *<br>+ * \retval non-NULL success<br>+ * \retval NULL failure<br>+ */<br>+static struct softmix_remb_collector *remb_collector_alloc(void)<br>+{<br>+       struct softmix_remb_collector *collector;<br>+<br>+ collector = ao2_alloc_options(sizeof(*collector), NULL, AO2_ALLOC_OPT_LOCK_NOLOCK);<br>+  if (!collector) {<br>+            return NULL;<br>+ }<br>+<br>+ collector->frame.frametype = AST_FRAME_RTCP;<br>+      collector->frame.subclass.integer = AST_RTP_RTCP_PSFB;<br>+    collector->feedback.fmt = AST_RTP_RTCP_FMT_REMB;<br>+  collector->frame.data.ptr = &collector->feedback;<br>+  collector->frame.datalen = sizeof(collector->feedback);<br>+<br>+     return collector;<br>+}<br>+<br>+/*!<br>+ * \brief Setup REMB collection for a particular bridge stream and channel.<br>+ *<br>+ * \param bridge The bridge<br>+ * \param bridge_channel Channel that is collecting REMB information<br>+ * \param bridge_stream_position The slot in the bridge where source video comes from<br>+ */<br>+static void remb_enable_collection(struct ast_bridge *bridge, struct ast_bridge_channel *bridge_channel,<br>+      size_t bridge_stream_position)<br>+{<br>+   struct softmix_channel *sc = bridge_channel->tech_pvt;<br>+    struct softmix_bridge_data *softmix_data = bridge->tech_pvt;<br>+<br>+   if (!sc->remb_collector) {<br>+                sc->remb_collector = remb_collector_alloc();<br>+              if (!sc->remb_collector) {<br>+                        /* This is not fatal. Things will still continue to work but we won't<br>+                     * produce a REMB report to the sender.<br>+                       */<br>+                  return;<br>+              }<br>+    }<br>+<br>+ if (AST_VECTOR_REPLACE(&softmix_data->remb_collectors, bridge_stream_position, ao2_bump(sc->remb_collector))) {<br>+            ao2_ref(sc->remb_collector, -1);<br>   }<br> }<br> <br>@@ -1835,6 +2057,8 @@<br>  */<br> static void softmix_bridge_stream_topology_changed(struct ast_bridge *bridge, struct ast_bridge_channel *bridge_channel)<br> {<br>+ struct softmix_bridge_data *softmix_data = bridge->tech_pvt;<br>+      struct softmix_channel *sc;<br>   struct ast_bridge_channel *participant;<br>       struct ast_vector_int media_types;<br>    int nths[AST_MEDIA_TYPE_END] = {0};<br>@@ -1852,11 +2076,22 @@<br> <br>       AST_VECTOR_INIT(&media_types, AST_MEDIA_TYPE_END);<br> <br>+    /* The bridge stream identifiers may change, so reset the mapping for them.<br>+   * When channels end up getting added back in they'll reuse their existing<br>+        * collector and won't need to allocate a new one (unless they were just added).<br>+  */<br>+  AST_VECTOR_RESET(&softmix_data->remb_collectors, ao2_cleanup);<br>+<br>      /* First traversal: re-initialize all of the participants' stream maps */<br>         AST_LIST_TRAVERSE(&bridge->channels, participant, entry) {<br>             ast_bridge_channel_lock(participant);<br>+<br>              AST_VECTOR_RESET(&participant->stream_map.to_channel, AST_VECTOR_ELEM_CLEANUP_NOOP);<br>           AST_VECTOR_RESET(&participant->stream_map.to_bridge, AST_VECTOR_ELEM_CLEANUP_NOOP);<br>+<br>+                sc = participant->tech_pvt;<br>+               AST_VECTOR_RESET(&sc->video_sources, AST_VECTOR_ELEM_CLEANUP_NOOP);<br>+<br>                 ast_bridge_channel_unlock(participant);<br>       }<br> <br>@@ -1897,7 +2132,12 @@<br>                  if (is_video_source(stream)) {<br>                                AST_VECTOR_APPEND(&media_types, AST_MEDIA_TYPE_VIDEO);<br>                            AST_VECTOR_REPLACE(&participant->stream_map.to_bridge, i, AST_VECTOR_SIZE(&media_types) - 1);<br>-                             AST_VECTOR_REPLACE(&participant->stream_map.to_channel, AST_VECTOR_SIZE(&media_types) - 1, -1);<br>+                           /*<br>+                            * There are cases where we need to bidirectionally send frames, such as for REMB reports<br>+                             * so we also map back to the channel.<br>+                                */<br>+                          AST_VECTOR_REPLACE(&participant->stream_map.to_channel, AST_VECTOR_SIZE(&media_types) - 1, i);<br>+                            remb_enable_collection(bridge, participant, AST_VECTOR_SIZE(&media_types) - 1);<br>                           /*<br>                             * Unlock the channel and participant to prevent<br>                               * potential deadlock in map_source_to_destinations().<br>diff --git a/bridges/bridge_softmix/include/bridge_softmix_internal.h b/bridges/bridge_softmix/include/bridge_softmix_internal.h<br>index f842acb..3aa9091 100644<br>--- a/bridges/bridge_softmix/include/bridge_softmix_internal.h<br>+++ b/bridges/bridge_softmix/include/bridge_softmix_internal.h<br>@@ -50,6 +50,8 @@<br> #include "asterisk/astobj2.h"<br> #include "asterisk/timing.h"<br> #include "asterisk/translate.h"<br>+#include "asterisk/rtp_engine.h"<br>+#include "asterisk/vector.h"<br> <br> #ifdef BINAURAL_RENDERING<br> #include <fftw3.h><br>@@ -124,6 +126,8 @@<br>        int energy_average;<br> };<br> <br>+struct softmix_remb_collector;<br>+<br> /*! \brief Structure which contains per-channel mixing information */<br> struct softmix_channel {<br>    /*! Lock to protect this structure */<br>@@ -169,6 +173,12 @@<br>   struct video_follow_talker_data video_talker;<br>         /*! The ideal stream topology for the channel */<br>      struct ast_stream_topology *topology;<br>+        /*! The latest REMB report from this participant */<br>+  struct ast_rtp_rtcp_feedback_remb remb;<br>+      /*! The REMB collector for this channel, collects REMB from all video receivers */<br>+   struct softmix_remb_collector *remb_collector;<br>+       /*! The bridge streams which are feeding us video sources */<br>+ AST_VECTOR(, int) video_sources;<br> };<br> <br> struct softmix_bridge_data {<br>@@ -202,6 +212,10 @@<br>         unsigned int binaural_init;<br>   /*! The last time a video update was sent into the bridge */<br>  struct timeval last_video_update;<br>+    /*! The last time a REMB frame was sent to each source of video */<br>+   struct timeval last_remb_update;<br>+     /*! Per-bridge stream REMB collectors, which flow back to video source */<br>+    AST_VECTOR(, struct softmix_remb_collector *) remb_collectors;<br> };<br> <br> struct softmix_mixing_array {<br>diff --git a/configs/samples/confbridge.conf.sample b/configs/samples/confbridge.conf.sample<br>index 4028593..8b276cd 100644<br>--- a/configs/samples/confbridge.conf.sample<br>+++ b/configs/samples/confbridge.conf.sample<br>@@ -239,6 +239,10 @@<br>                            ; A REMB frame contains receiver estimated maximum bitrate information. By creating a combined<br>                            ; frame and sending it to the sources of video the sender can be influenced on what bitrate<br>                            ; they choose allowing a better experience for the receivers. This defaults to 0, or disabled.<br>+;remb_behavior=average     ; How the combined REMB report for an SFU video bridge is constructed. If set to "average" then<br>+                           ; the estimated maximum bitrate of each receiver is used to construct an average bitrate. If<br>+                           ; set to "lowest" the lowest maximum bitrate is forwarded to the sender. If set to "highest"<br>+                           ; the highest maximum bitrate is forwarded to the sender. This defaults to "average".<br> <br> ; All sounds in the conference are customizable using the bridge profile options below.<br> ; Simply state the option followed by the filename or full path of the filename after<br>diff --git a/include/asterisk/bridge.h b/include/asterisk/bridge.h<br>index c96cefb..3584085 100644<br>--- a/include/asterisk/bridge.h<br>+++ b/include/asterisk/bridge.h<br>@@ -126,6 +126,24 @@<br>    struct ast_channel *chan_old_vsrc;<br> };<br> <br>+/*! \brief REMB report behaviors */<br>+enum ast_bridge_video_sfu_remb_behavior {<br>+ /*! The average of all reports is sent to the sender */<br>+      AST_BRIDGE_VIDEO_SFU_REMB_AVERAGE = 0,<br>+       /*! The lowest reported bitrate is forwarded to the sender */<br>+        AST_BRIDGE_VIDEO_SFU_REMB_LOWEST,<br>+    /*! The highest reported bitrate is forwarded to the sender */<br>+       AST_BRIDGE_VIDEO_SFU_REMB_HIGHEST,<br>+};<br>+<br>+/*! \brief This is used for selective forwarding unit configuration */<br>+struct ast_bridge_video_sfu_data {<br>+     /*! The interval at which a REMB report is generated and sent */<br>+     unsigned int remb_send_interval;<br>+     /*! How the combined REMB report is generated */<br>+     enum ast_bridge_video_sfu_remb_behavior remb_behavior;<br>+};<br>+<br> /*! \brief Data structure that defines a video source mode */<br> struct ast_bridge_video_mode {<br>       enum ast_bridge_video_mode_type mode;<br>@@ -133,9 +151,10 @@<br>   union {<br>               struct ast_bridge_video_single_src_data single_src_data;<br>              struct ast_bridge_video_talker_src_data talker_src_data;<br>+             struct ast_bridge_video_sfu_data sfu_data;<br>    } mode_data;<br>+ /*! The minimum interval between video updates */<br>     unsigned int video_update_discard;<br>-   unsigned int remb_send_interval;<br> };<br> <br> /*!<br>@@ -917,10 +936,22 @@<br>  *<br>  * \param bridge Bridge to set the REMB send interval on<br>  * \param remb_send_interval The REMB send interval<br>+ *<br>+ * \note This can only be called when the bridge has been set to the SFU video mode.<br>  */<br> void ast_bridge_set_remb_send_interval(struct ast_bridge *bridge, unsigned int remb_send_interval);<br> <br> /*!<br>+ * \brief Set the REMB report generation behavior on a bridge<br>+ *<br>+ * \param bridge Bridge to set the REMB behavior on<br>+ * \param behavior How REMB reports are generated<br>+ *<br>+ * \note This can only be called when the bridge has been set to the SFU video mode.<br>+ */<br>+void ast_brige_set_remb_behavior(struct ast_bridge *bridge, enum ast_bridge_video_sfu_remb_behavior behavior);<br>+<br>+/*!<br>  * \brief Update information about talker energy for talker src video mode.<br>  */<br> void ast_bridge_update_talker_src_video_mode(struct ast_bridge *bridge, struct ast_channel *chan, int talker_energy, int is_keyfame);<br>diff --git a/main/bridge.c b/main/bridge.c<br>index 48cfa54..d76c71e 100644<br>--- a/main/bridge.c<br>+++ b/main/bridge.c<br>@@ -3834,8 +3834,19 @@<br> <br> void ast_bridge_set_remb_send_interval(struct ast_bridge *bridge, unsigned int remb_send_interval)<br> {<br>+ ast_assert(bridge->softmix.video_mode.mode == AST_BRIDGE_VIDEO_MODE_SFU);<br>+<br>       ast_bridge_lock(bridge);<br>-     bridge->softmix.video_mode.remb_send_interval = remb_send_interval;<br>+       bridge->softmix.video_mode.mode_data.sfu_data.remb_send_interval = remb_send_interval;<br>+    ast_bridge_unlock(bridge);<br>+}<br>+<br>+void ast_brige_set_remb_behavior(struct ast_bridge *bridge, enum ast_bridge_video_sfu_remb_behavior behavior)<br>+{<br>+        ast_assert(bridge->softmix.video_mode.mode == AST_BRIDGE_VIDEO_MODE_SFU);<br>+<br>+      ast_bridge_lock(bridge);<br>+     bridge->softmix.video_mode.mode_data.sfu_data.remb_behavior = behavior;<br>    ast_bridge_unlock(bridge);<br> }<br> <br></pre><p>To view, visit <a href="https://gerrit.asterisk.org/8782">change 8782</a>. To unsubscribe, visit <a href="https://gerrit.asterisk.org/settings">settings</a>.</p><div itemscope itemtype="http://schema.org/EmailMessage"><div itemscope itemprop="action" itemtype="http://schema.org/ViewAction"><link itemprop="url" href="https://gerrit.asterisk.org/8782"/><meta itemprop="name" content="View Change"/></div></div>

<div style="display:none"> Gerrit-Project: asterisk </div>
<div style="display:none"> Gerrit-Branch: 15 </div>
<div style="display:none"> Gerrit-MessageType: merged </div>
<div style="display:none"> Gerrit-Change-Id: I9eafe4e7c1f72d67074a8d6acb26bfcf19322b66 </div>
<div style="display:none"> Gerrit-Change-Number: 8782 </div>
<div style="display:none"> Gerrit-PatchSet: 3 </div>
<div style="display:none"> Gerrit-Owner: Joshua Colp <jcolp@digium.com> </div>
<div style="display:none"> Gerrit-Reviewer: George Joseph <gjoseph@digium.com> </div>
<div style="display:none"> Gerrit-Reviewer: Jenkins2 </div>
<div style="display:none"> Gerrit-Reviewer: Matthew Fredrickson <creslin@digium.com> </div>
<div style="display:none"> Gerrit-Reviewer: Richard Mudgett <rmudgett@digium.com> </div>