Transmit power level in a channel is determined based on the dfs region.
To support regulatory rules dfs region should be configured to device during
set channel request. Also antenna gain values are taken from the mac80211
channel parameters instead of fixed values.
Signed-off-by: Prameela Rani Garnepudi <prameela.j04cs@gmail.com>
Signed-off-by: Kalle Valo <kvalo@codeaurora.org>
 
        status = rsi_band_check(common);
        if (!status)
-               status = rsi_set_channel(adapter->priv, channel);
+               status = rsi_set_channel(adapter->priv, curchan);
 
        if (bss->assoc) {
                if (common->hw_data_qs_blocked &&
        return 0;       
 }
 
+static void rsi_reg_notify(struct wiphy *wiphy,
+                          struct regulatory_request *request)
+{
+       struct ieee80211_supported_band *sband;
+       struct ieee80211_channel *ch;
+       struct ieee80211_hw *hw = wiphy_to_ieee80211_hw(wiphy);
+       struct rsi_hw * adapter = hw->priv; 
+       int i;
+
+       sband = wiphy->bands[NL80211_BAND_5GHZ];
+       
+       for (i = 0; i < sband->n_channels; i++) {
+               ch = &sband->channels[i];
+               if (ch->flags & IEEE80211_CHAN_DISABLED)
+                       continue;
+
+               if (ch->flags & IEEE80211_CHAN_RADAR)
+                       ch->flags |= IEEE80211_CHAN_NO_IR;
+       }
+       
+       rsi_dbg(INFO_ZONE,
+               "country = %s dfs_region = %d\n",
+               request->alpha2, request->dfs_region);
+       adapter->dfs_region = request->dfs_region;
+}
+
 static struct ieee80211_ops mac80211_ops = {
        .tx = rsi_mac80211_tx,
        .start = rsi_mac80211_start,
        wiphy->bands[NL80211_BAND_5GHZ] =
                &adapter->sbands[NL80211_BAND_5GHZ];
 
+       wiphy->reg_notifier = rsi_reg_notify;
+
        status = ieee80211_register_hw(hw);
        if (status)
                return status;
 
  *
  * Return: 0 on success, corresponding error code on failure.
  */
-int rsi_set_channel(struct rsi_common *common, u16 channel)
+int rsi_set_channel(struct rsi_common *common,
+                   struct ieee80211_channel *channel)
 {
        struct sk_buff *skb = NULL;
        struct rsi_mac_frame *mgmt_frame;
                return -ENOMEM;
        }
 
+       if (!channel) {
+               dev_kfree_skb(skb);
+               return 0;
+       }
        memset(skb->data, 0, FRAME_DESC_SZ);
        mgmt_frame = (struct rsi_mac_frame *)skb->data;
 
        mgmt_frame->desc_word[0] = cpu_to_le16(RSI_WIFI_MGMT_Q << 12);
        mgmt_frame->desc_word[1] = cpu_to_le16(SCAN_REQUEST);
-       mgmt_frame->desc_word[4] = cpu_to_le16(channel);
+       mgmt_frame->desc_word[4] = cpu_to_le16(channel->hw_value);
+
+       mgmt_frame->desc_word[4] |=
+               cpu_to_le16(((char)(channel->max_antenna_gain)) << 8);
+       mgmt_frame->desc_word[5] =
+               cpu_to_le16((char)(channel->max_antenna_gain));
 
        mgmt_frame->desc_word[7] = cpu_to_le16(PUT_BBP_RESET |
                                               BBP_REG_WRITE |
                                               (RSI_RF_TYPE << 4));
 
-       mgmt_frame->desc_word[5] = cpu_to_le16(0x01);
-       mgmt_frame->desc_word[6] = cpu_to_le16(0x12);
+       if (!(channel->flags & IEEE80211_CHAN_NO_IR) &&
+              !(channel->flags & IEEE80211_CHAN_RADAR)) {
+               if (common->tx_power < channel->max_power)
+                       mgmt_frame->desc_word[6] = cpu_to_le16(common->tx_power);
+               else
+                       mgmt_frame->desc_word[6] = cpu_to_le16(channel->max_power);
+       }
+       mgmt_frame->desc_word[7] = cpu_to_le16(common->priv->dfs_region);
 
        if (common->channel_width == BW_40MHZ)
                mgmt_frame->desc_word[5] |= cpu_to_le16(0x1 << 8);
 
-       common->channel = channel;
+       common->channel = channel->hw_value;
 
        skb_put(skb, FRAME_DESC_SZ);
 
 
        struct rsi_debugfs *dfsentry;
        u8 num_debugfs_entries;
 #endif
+       u8 dfs_region;
        void *rsi_dev;
        int (*host_intf_read_pkt)(struct rsi_hw *adapter, u8 *pkt, u32 len);
        int (*host_intf_write_pkt)(struct rsi_hw *adapter, u8 *pkt, u32 len);
 
                                      u16 ssn, u8 buf_size, u8 event);
 int rsi_hal_load_key(struct rsi_common *common, u8 *data, u16 key_len,
                     u8 key_type, u8 key_id, u32 cipher);
-int rsi_set_channel(struct rsi_common *common, u16 chno);
+int rsi_set_channel(struct rsi_common *common,
+                   struct ieee80211_channel *channel);
 int rsi_send_block_unblock_frame(struct rsi_common *common, bool event);
 void rsi_inform_bss_status(struct rsi_common *common, u8 status,
                           const u8 *bssid, u8 qos_enable, u16 aid);