* @dests: array keeping number of reachable destinations per bearer
  * @primary_bearer: a bearer having links to all broadcast destinations, if any
  * @bcast_support: indicates if primary bearer, if any, supports broadcast
+ * @force_bcast: forces broadcast for multicast traffic
  * @rcast_support: indicates if all peer nodes support replicast
+ * @force_rcast: forces replicast for multicast traffic
  * @rc_ratio: dest count as percentage of cluster size where send method changes
  * @bc_threshold: calculated from rc_ratio; if dests > threshold use broadcast
  */
        int dests[MAX_BEARERS];
        int primary_bearer;
        bool bcast_support;
+       bool force_bcast;
        bool rcast_support;
+       bool force_rcast;
        int rc_ratio;
        int bc_threshold;
 };
        return 0;
 }
 
+static int tipc_bc_link_set_broadcast_mode(struct net *net, u32 bc_mode)
+{
+       struct tipc_bc_base *bb = tipc_bc_base(net);
+
+       switch (bc_mode) {
+       case BCLINK_MODE_BCAST:
+               if (!bb->bcast_support)
+                       return -ENOPROTOOPT;
+
+               bb->force_bcast = true;
+               bb->force_rcast = false;
+               break;
+       case BCLINK_MODE_RCAST:
+               if (!bb->rcast_support)
+                       return -ENOPROTOOPT;
+
+               bb->force_bcast = false;
+               bb->force_rcast = true;
+               break;
+       case BCLINK_MODE_SEL:
+               if (!bb->bcast_support || !bb->rcast_support)
+                       return -ENOPROTOOPT;
+
+               bb->force_bcast = false;
+               bb->force_rcast = false;
+               break;
+       default:
+               return -EINVAL;
+       }
+
+       return 0;
+}
+
+static int tipc_bc_link_set_broadcast_ratio(struct net *net, u32 bc_ratio)
+{
+       struct tipc_bc_base *bb = tipc_bc_base(net);
+
+       if (!bb->bcast_support || !bb->rcast_support)
+               return -ENOPROTOOPT;
+
+       if (bc_ratio > 100 || bc_ratio <= 0)
+               return -EINVAL;
+
+       bb->rc_ratio = bc_ratio;
+       tipc_bcast_lock(net);
+       tipc_bcbase_calc_bc_threshold(net);
+       tipc_bcast_unlock(net);
+
+       return 0;
+}
+
 int tipc_nl_bc_link_set(struct net *net, struct nlattr *attrs[])
 {
        int err;
        u32 win;
+       u32 bc_mode;
+       u32 bc_ratio;
        struct nlattr *props[TIPC_NLA_PROP_MAX + 1];
 
        if (!attrs[TIPC_NLA_LINK_PROP])
        if (err)
                return err;
 
-       if (!props[TIPC_NLA_PROP_WIN])
+       if (!props[TIPC_NLA_PROP_WIN] &&
+           !props[TIPC_NLA_PROP_BROADCAST] &&
+           !props[TIPC_NLA_PROP_BROADCAST_RATIO]) {
                return -EOPNOTSUPP;
+       }
+
+       if (props[TIPC_NLA_PROP_BROADCAST]) {
+               bc_mode = nla_get_u32(props[TIPC_NLA_PROP_BROADCAST]);
+               err = tipc_bc_link_set_broadcast_mode(net, bc_mode);
+       }
 
-       win = nla_get_u32(props[TIPC_NLA_PROP_WIN]);
+       if (!err && props[TIPC_NLA_PROP_BROADCAST_RATIO]) {
+               bc_ratio = nla_get_u32(props[TIPC_NLA_PROP_BROADCAST_RATIO]);
+               err = tipc_bc_link_set_broadcast_ratio(net, bc_ratio);
+       }
 
-       return tipc_bc_link_set_queue_limits(net, win);
+       if (!err && props[TIPC_NLA_PROP_WIN]) {
+               win = nla_get_u32(props[TIPC_NLA_PROP_WIN]);
+               err = tipc_bc_link_set_queue_limits(net, win);
+       }
+
+       return err;
 }
 
 int tipc_bcast_init(struct net *net)
                goto enomem;
        bb->link = l;
        tn->bcl = l;
-       bb->rc_ratio = 25;
+       bb->rc_ratio = 10;
        bb->rcast_support = true;
        return 0;
 enomem:
        nl->remote = 0;
        nl->local = false;
 }
+
+u32 tipc_bcast_get_broadcast_mode(struct net *net)
+{
+       struct tipc_bc_base *bb = tipc_bc_base(net);
+
+       if (bb->force_bcast)
+               return BCLINK_MODE_BCAST;
+
+       if (bb->force_rcast)
+               return BCLINK_MODE_RCAST;
+
+       if (bb->bcast_support && bb->rcast_support)
+               return BCLINK_MODE_SEL;
+
+       return 0;
+}
+
+u32 tipc_bcast_get_broadcast_ratio(struct net *net)
+{
+       struct tipc_bc_base *bb = tipc_bc_base(net);
+
+       return bb->rc_ratio;
+}
 
        struct nlattr *attrs;
        struct nlattr *prop;
        struct tipc_net *tn = net_generic(net, tipc_net_id);
+       u32 bc_mode = tipc_bcast_get_broadcast_mode(net);
+       u32 bc_ratio = tipc_bcast_get_broadcast_ratio(net);
        struct tipc_link *bcl = tn->bcl;
 
        if (!bcl)
                goto attr_msg_full;
        if (nla_put_u32(msg->skb, TIPC_NLA_PROP_WIN, bcl->window))
                goto prop_msg_full;
+       if (nla_put_u32(msg->skb, TIPC_NLA_PROP_BROADCAST, bc_mode))
+               goto prop_msg_full;
+       if (bc_mode & BCLINK_MODE_SEL)
+               if (nla_put_u32(msg->skb, TIPC_NLA_PROP_BROADCAST_RATIO,
+                               bc_ratio))
+                       goto prop_msg_full;
        nla_nest_end(msg->skb, prop);
 
        err = __tipc_nl_add_bc_link_stat(msg->skb, &bcl->stats);