#define CCK_GROUP __CCK_GROUP(CCK_GROUP_SHIFT)
 
+#define OFDM_DURATION(_bitrate)                                \
+       (1000 * (16 /* SIFS + signal ext */ +           \
+        16 /* T_PREAMBLE */ +                          \
+        4 /* T_SIGNAL */ +                             \
+        4 * (((16 + 80 * (AVG_PKT_SIZE + 4) + 6) /     \
+             ((_bitrate) * 4)))))
+
+#define OFDM_DURATION_LIST(_s)                         \
+       OFDM_DURATION(60) >> _s,                        \
+       OFDM_DURATION(90) >> _s,                        \
+       OFDM_DURATION(120) >> _s,                       \
+       OFDM_DURATION(180) >> _s,                       \
+       OFDM_DURATION(240) >> _s,                       \
+       OFDM_DURATION(360) >> _s,                       \
+       OFDM_DURATION(480) >> _s,                       \
+       OFDM_DURATION(540) >> _s
+
+#define __OFDM_GROUP(_s)                               \
+       [MINSTREL_OFDM_GROUP] = {                       \
+               .streams = 1,                           \
+               .flags = 0,                             \
+               .shift = _s,                            \
+               .duration = {                           \
+                       OFDM_DURATION_LIST(_s),         \
+               }                                       \
+       }
+
+#define OFDM_GROUP_SHIFT                               \
+       GROUP_SHIFT(OFDM_DURATION(60))
+
+#define OFDM_GROUP __OFDM_GROUP(OFDM_GROUP_SHIFT)
+
 
 static bool minstrel_vht_only = true;
 module_param(minstrel_vht_only, bool, 0644);
        MCS_GROUP(4, 1, BW_40),
 
        CCK_GROUP,
+       OFDM_GROUP,
 
        VHT_GROUP(1, 0, BW_20),
        VHT_GROUP(2, 0, BW_20),
        VHT_GROUP(4, 1, BW_80),
 };
 
+const s16 minstrel_cck_bitrates[4] = { 10, 20, 55, 110 };
+const s16 minstrel_ofdm_bitrates[8] = { 60, 90, 120, 180, 240, 360, 480, 540 };
 static u8 sample_table[SAMPLE_COLUMNS][MCS_GROUP_RATES] __read_mostly;
 
 static void
        return 0x3ff & ~mask;
 }
 
+static bool
+minstrel_ht_is_legacy_group(int group)
+{
+       return group == MINSTREL_CCK_GROUP ||
+              group == MINSTREL_OFDM_GROUP;
+}
+
 /*
  * Look up an MCS group index based on mac80211 rate information
  */
        if (rate->flags & IEEE80211_TX_RC_MCS) {
                group = minstrel_ht_get_group_idx(rate);
                idx = rate->idx % 8;
-       } else if (rate->flags & IEEE80211_TX_RC_VHT_MCS) {
+               goto out;
+       }
+
+       if (rate->flags & IEEE80211_TX_RC_VHT_MCS) {
                group = minstrel_vht_get_group_idx(rate);
                idx = ieee80211_rate_get_vht_mcs(rate);
-       } else {
-               group = MINSTREL_CCK_GROUP;
+               goto out;
+       }
 
-               for (idx = 0; idx < ARRAY_SIZE(mp->cck_rates); idx++)
-                       if (rate->idx == mp->cck_rates[idx])
-                               break;
+       group = MINSTREL_CCK_GROUP;
+       for (idx = 0; idx < ARRAY_SIZE(mp->cck_rates); idx++) {
+               if (rate->idx != mp->cck_rates[idx])
+                       continue;
 
                /* short preamble */
                if ((mi->supported[group] & BIT(idx + 4)) &&
                    (rate->flags & IEEE80211_TX_RC_USE_SHORT_PREAMBLE))
-                       idx += 4;
+                               idx += 4;
+               goto out;
        }
+
+       group = MINSTREL_OFDM_GROUP;
+       for (idx = 0; idx < ARRAY_SIZE(mp->ofdm_rates[0]); idx++)
+               if (rate->idx == mp->ofdm_rates[mi->band][idx])
+                       goto out;
+
+       idx = 0;
+out:
        return &mi->groups[group].rates[idx];
 }
 
        if (prob_avg < MINSTREL_FRAC(10, 100))
                return 0;
 
-       if (group == MINSTREL_CCK_GROUP)
+       if (minstrel_ht_is_legacy_group(group))
                overhead = mi->overhead_legacy;
        else
                ampdu_len = minstrel_ht_avg_ampdu_len(mi);
        /* if max_tp_rate[0] is from MCS_GROUP max_prob_rate get selected from
         * MCS_GROUP as well as CCK_GROUP rates do not allow aggregation */
        max_tp_group = mi->max_tp_rate[0] / MCS_GROUP_RATES;
-       if((index / MCS_GROUP_RATES == MINSTREL_CCK_GROUP) &&
-           (max_tp_group != MINSTREL_CCK_GROUP))
+       if (minstrel_ht_is_legacy_group(index / MCS_GROUP_RATES) &&
+           !minstrel_ht_is_legacy_group(max_tp_group))
                return;
 
        max_gpr_group = mg->max_group_prob_rate / MCS_GROUP_RATES;
 static void
 minstrel_ht_assign_best_tp_rates(struct minstrel_ht_sta *mi,
                                 u16 tmp_mcs_tp_rate[MAX_THR_RATES],
-                                u16 tmp_cck_tp_rate[MAX_THR_RATES])
+                                u16 tmp_legacy_tp_rate[MAX_THR_RATES])
 {
        unsigned int tmp_group, tmp_idx, tmp_cck_tp, tmp_mcs_tp, tmp_prob;
        int i;
 
-       tmp_group = tmp_cck_tp_rate[0] / MCS_GROUP_RATES;
-       tmp_idx = tmp_cck_tp_rate[0] % MCS_GROUP_RATES;
+       tmp_group = tmp_legacy_tp_rate[0] / MCS_GROUP_RATES;
+       tmp_idx = tmp_legacy_tp_rate[0] % MCS_GROUP_RATES;
        tmp_prob = mi->groups[tmp_group].rates[tmp_idx].prob_avg;
        tmp_cck_tp = minstrel_ht_get_tp_avg(mi, tmp_group, tmp_idx, tmp_prob);
 
 
        if (tmp_cck_tp > tmp_mcs_tp) {
                for(i = 0; i < MAX_THR_RATES; i++) {
-                       minstrel_ht_sort_best_tp_rates(mi, tmp_cck_tp_rate[i],
+                       minstrel_ht_sort_best_tp_rates(mi, tmp_legacy_tp_rate[i],
                                                       tmp_mcs_tp_rate);
                }
        }
        int tmp_max_streams, group, tmp_idx, tmp_prob;
        int tmp_tp = 0;
 
+       if (!mi->sta->ht_cap.ht_supported)
+               return;
+
        tmp_max_streams = minstrel_mcs_groups[mi->max_tp_rate[0] /
                          MCS_GROUP_RATES].streams;
        for (group = 0; group < ARRAY_SIZE(minstrel_mcs_groups); group++) {
        struct minstrel_rate_stats *mrs;
        int group, i, j, cur_prob;
        u16 tmp_mcs_tp_rate[MAX_THR_RATES], tmp_group_tp_rate[MAX_THR_RATES];
-       u16 tmp_cck_tp_rate[MAX_THR_RATES], index;
+       u16 tmp_legacy_tp_rate[MAX_THR_RATES], index;
+       bool ht_supported = mi->sta->ht_cap.ht_supported;
 
        mi->sample_mode = MINSTREL_SAMPLE_IDLE;
 
        mi->sample_count = 0;
 
        memset(tmp_mcs_tp_rate, 0, sizeof(tmp_mcs_tp_rate));
-       memset(tmp_cck_tp_rate, 0, sizeof(tmp_cck_tp_rate));
+       memset(tmp_legacy_tp_rate, 0, sizeof(tmp_legacy_tp_rate));
        if (mi->supported[MINSTREL_CCK_GROUP])
-               for (j = 0; j < ARRAY_SIZE(tmp_cck_tp_rate); j++)
-                       tmp_cck_tp_rate[j] = MINSTREL_CCK_GROUP * MCS_GROUP_RATES;
+               for (j = 0; j < ARRAY_SIZE(tmp_legacy_tp_rate); j++)
+                       tmp_legacy_tp_rate[j] = MINSTREL_CCK_GROUP * MCS_GROUP_RATES;
+       else if (mi->supported[MINSTREL_OFDM_GROUP])
+               for (j = 0; j < ARRAY_SIZE(tmp_legacy_tp_rate); j++)
+                       tmp_legacy_tp_rate[j] = MINSTREL_OFDM_GROUP * MCS_GROUP_RATES;
 
        if (mi->supported[MINSTREL_VHT_GROUP_0])
                index = MINSTREL_VHT_GROUP_0 * MCS_GROUP_RATES;
-       else
+       else if (ht_supported)
                index = MINSTREL_HT_GROUP_0 * MCS_GROUP_RATES;
+       else if (mi->supported[MINSTREL_CCK_GROUP])
+               index = MINSTREL_CCK_GROUP * MCS_GROUP_RATES;
+       else
+               index = MINSTREL_OFDM_GROUP * MCS_GROUP_RATES;
 
        for (j = 0; j < ARRAY_SIZE(tmp_mcs_tp_rate); j++)
                tmp_mcs_tp_rate[j] = index;
 
        /* Find best rate sets within all MCS groups*/
        for (group = 0; group < ARRAY_SIZE(minstrel_mcs_groups); group++) {
+               u16 *tp_rate = tmp_mcs_tp_rate;
 
                mg = &mi->groups[group];
                if (!mi->supported[group])
                for(j = 0; j < MAX_THR_RATES; j++)
                        tmp_group_tp_rate[j] = MCS_GROUP_RATES * group;
 
+               if (group == MINSTREL_CCK_GROUP && ht_supported)
+                       tp_rate = tmp_legacy_tp_rate;
+
                for (i = 0; i < MCS_GROUP_RATES; i++) {
                        if (!(mi->supported[group] & BIT(i)))
                                continue;
                                continue;
 
                        /* Find max throughput rate set */
-                       if (group != MINSTREL_CCK_GROUP) {
-                               minstrel_ht_sort_best_tp_rates(mi, index,
-                                                              tmp_mcs_tp_rate);
-                       } else if (group == MINSTREL_CCK_GROUP) {
-                               minstrel_ht_sort_best_tp_rates(mi, index,
-                                                              tmp_cck_tp_rate);
-                       }
+                       minstrel_ht_sort_best_tp_rates(mi, index, tp_rate);
 
                        /* Find max throughput rate set within a group */
                        minstrel_ht_sort_best_tp_rates(mi, index,
        }
 
        /* Assign new rate set per sta */
-       minstrel_ht_assign_best_tp_rates(mi, tmp_mcs_tp_rate, tmp_cck_tp_rate);
+       minstrel_ht_assign_best_tp_rates(mi, tmp_mcs_tp_rate,
+                                        tmp_legacy_tp_rate);
        memcpy(mi->max_tp_rate, tmp_mcs_tp_rate, sizeof(mi->max_tp_rate));
 
        /* Try to increase robustness of max_prob_rate*/
 }
 
 static bool
-minstrel_ht_txstat_valid(struct minstrel_priv *mp, struct ieee80211_tx_rate *rate)
+minstrel_ht_txstat_valid(struct minstrel_priv *mp, struct minstrel_ht_sta *mi,
+                        struct ieee80211_tx_rate *rate)
 {
+       int i;
+
        if (rate->idx < 0)
                return false;
 
            rate->flags & IEEE80211_TX_RC_VHT_MCS)
                return true;
 
-       return rate->idx == mp->cck_rates[0] ||
-              rate->idx == mp->cck_rates[1] ||
-              rate->idx == mp->cck_rates[2] ||
-              rate->idx == mp->cck_rates[3];
+       for (i = 0; i < ARRAY_SIZE(mp->cck_rates); i++)
+               if (rate->idx == mp->cck_rates[i])
+                       return true;
+
+       for (i = 0; i < ARRAY_SIZE(mp->ofdm_rates[0]); i++)
+               if (rate->idx == mp->ofdm_rates[mi->band][i])
+                       return true;
+
+       return false;
 }
 
 static void
        bool sample_status = false;
        int i;
 
-       if (!msp->is_ht)
-               return mac80211_minstrel.tx_status_ext(priv, sband,
-                                                      &msp->legacy, st);
-
-
        /* This packet was aggregated but doesn't carry status info */
        if ((info->flags & IEEE80211_TX_CTL_AMPDU) &&
            !(info->flags & IEEE80211_TX_STAT_AMPDU))
        if (mi->sample_mode != MINSTREL_SAMPLE_IDLE)
                rate_sample = minstrel_get_ratestats(mi, mi->sample_rate);
 
-       last = !minstrel_ht_txstat_valid(mp, &ar[0]);
+       last = !minstrel_ht_txstat_valid(mp, mi, &ar[0]);
        for (i = 0; !last; i++) {
                last = (i == IEEE80211_TX_MAX_RATES - 1) ||
-                      !minstrel_ht_txstat_valid(mp, &ar[i + 1]);
+                      !minstrel_ht_txstat_valid(mp, mi, &ar[i + 1]);
 
                rate = minstrel_ht_get_stats(mp, mi, &ar[i]);
                if (rate == rate_sample)
        ctime += (t_slot * cw) >> 1;
        cw = min((cw << 1) | 1, mp->cw_max);
 
-       if (index / MCS_GROUP_RATES == MINSTREL_CCK_GROUP) {
+       if (minstrel_ht_is_legacy_group(index / MCS_GROUP_RATES)) {
                overhead = mi->overhead_legacy;
                overhead_rtscts = mi->overhead_legacy_rtscts;
        } else {
 minstrel_ht_set_rate(struct minstrel_priv *mp, struct minstrel_ht_sta *mi,
                      struct ieee80211_sta_rates *ratetbl, int offset, int index)
 {
-       const struct mcs_group *group = &minstrel_mcs_groups[index / MCS_GROUP_RATES];
+       int group_idx = index / MCS_GROUP_RATES;
+       const struct mcs_group *group = &minstrel_mcs_groups[group_idx];
        struct minstrel_rate_stats *mrs;
        u8 idx;
        u16 flags = group->flags;
                ratetbl->rate[offset].count_rts = mrs->retry_count_rtscts;
        }
 
-       if (index / MCS_GROUP_RATES == MINSTREL_CCK_GROUP)
+       index %= MCS_GROUP_RATES;
+       if (group_idx == MINSTREL_CCK_GROUP)
                idx = mp->cck_rates[index % ARRAY_SIZE(mp->cck_rates)];
+       else if (group_idx == MINSTREL_OFDM_GROUP)
+               idx = mp->ofdm_rates[mi->band][index %
+                                              ARRAY_SIZE(mp->ofdm_rates[0])];
        else if (flags & IEEE80211_TX_RC_VHT_MCS)
                idx = ((group->streams - 1) << 4) |
-                     ((index % MCS_GROUP_RATES) & 0xF);
+                     (index & 0xF);
        else
-               idx = index % MCS_GROUP_RATES + (group->streams - 1) * 8;
+               idx = index + (group->streams - 1) * 8;
 
        /* enable RTS/CTS if needed:
         *  - if station is in dynamic SMPS (and streams > 1)
        struct minstrel_priv *mp = priv;
        int sample_idx;
 
-       if (!msp->is_ht)
-               return mac80211_minstrel.get_rate(priv, sta, &msp->legacy, txrc);
-
        if (!(info->flags & IEEE80211_TX_CTL_AMPDU) &&
-           mi->max_prob_rate / MCS_GROUP_RATES != MINSTREL_CCK_GROUP)
+           !minstrel_ht_is_legacy_group(mi->max_prob_rate / MCS_GROUP_RATES))
                minstrel_aggr_check(sta, txrc->skb);
 
        info->flags |= mi->tx_flags;
        if (sample_group == &minstrel_mcs_groups[MINSTREL_CCK_GROUP]) {
                int idx = sample_idx % ARRAY_SIZE(mp->cck_rates);
                rate->idx = mp->cck_rates[idx];
+       } else if (sample_group == &minstrel_mcs_groups[MINSTREL_OFDM_GROUP]) {
+               int idx = sample_idx % ARRAY_SIZE(mp->ofdm_rates[0]);
+               rate->idx = mp->ofdm_rates[mi->band][idx];
        } else if (sample_group->flags & IEEE80211_TX_RC_VHT_MCS) {
                ieee80211_rate_set_vht(rate, sample_idx % MCS_GROUP_RATES,
                                       sample_group->streams);
        if (sband->band != NL80211_BAND_2GHZ)
                return;
 
-       if (!ieee80211_hw_check(mp->hw, SUPPORTS_HT_CCK_RATES))
+       if (sta->ht_cap.ht_supported &&
+           !ieee80211_hw_check(mp->hw, SUPPORTS_HT_CCK_RATES))
                return;
 
        for (i = 0; i < 4; i++) {
-               if (!rate_supported(sta, sband->band, mp->cck_rates[i]))
+               if (mp->cck_rates[i] == 0xff ||
+                   !rate_supported(sta, sband->band, mp->cck_rates[i]))
                        continue;
 
                mi->supported[MINSTREL_CCK_GROUP] |= BIT(i);
        }
 }
 
+static void
+minstrel_ht_update_ofdm(struct minstrel_priv *mp, struct minstrel_ht_sta *mi,
+                       struct ieee80211_supported_band *sband,
+                       struct ieee80211_sta *sta)
+{
+       const u8 *rates;
+       int i;
+
+       if (sta->ht_cap.ht_supported)
+               return;
+
+       rates = mp->ofdm_rates[sband->band];
+       for (i = 0; i < ARRAY_SIZE(mp->ofdm_rates[0]); i++) {
+               if (rates[i] == 0xff ||
+                   !rate_supported(sta, sband->band, rates[i]))
+                       continue;
+
+               mi->supported[MINSTREL_OFDM_GROUP] |= BIT(i);
+       }
+}
+
 static void
 minstrel_ht_update_caps(void *priv, struct ieee80211_supported_band *sband,
                        struct cfg80211_chan_def *chandef,
-                        struct ieee80211_sta *sta, void *priv_sta)
+                       struct ieee80211_sta *sta, void *priv_sta)
 {
        struct minstrel_priv *mp = priv;
        struct minstrel_ht_sta_priv *msp = priv_sta;
        int stbc;
        int i;
 
-       /* fall back to the old minstrel for legacy stations */
-       if (!sta->ht_cap.ht_supported)
-               goto use_legacy;
-
        BUILD_BUG_ON(ARRAY_SIZE(minstrel_mcs_groups) != MINSTREL_GROUPS_NB);
 
        if (vht_cap->vht_supported)
        else
                use_vht = 0;
 
-       msp->is_ht = true;
        memset(mi, 0, sizeof(*mi));
 
        mi->sta = sta;
+       mi->band = sband->band;
        mi->last_stats_update = jiffies;
 
        ack_dur = ieee80211_frame_duration(sband->band, 10, 60, 1, 1, 0);
                int bw, nss;
 
                mi->supported[i] = 0;
-               if (i == MINSTREL_CCK_GROUP) {
-                       minstrel_ht_update_cck(mp, mi, sband, sta);
+               if (minstrel_ht_is_legacy_group(i))
                        continue;
-               }
 
                if (gflags & IEEE80211_TX_RC_SHORT_GI) {
                        if (gflags & IEEE80211_TX_RC_40_MHZ_WIDTH) {
                        n_supported++;
        }
 
-       if (!n_supported)
-               goto use_legacy;
+       minstrel_ht_update_cck(mp, mi, sband, sta);
+       minstrel_ht_update_ofdm(mp, mi, sband, sta);
 
        /* create an initial rate table with the lowest supported rates */
        minstrel_ht_update_stats(mp, mi, true);
        minstrel_ht_update_rates(mp, mi);
-
-       return;
-
-use_legacy:
-       msp->is_ht = false;
-       memset(&msp->legacy, 0, sizeof(msp->legacy));
-       msp->legacy.r = msp->ratelist;
-       msp->legacy.sample_table = msp->sample_table;
-       return mac80211_minstrel.rate_init(priv, sband, chandef, sta,
-                                          &msp->legacy);
 }
 
 static void
 }
 
 static void
-minstrel_ht_init_cck_rates(struct minstrel_priv *mp)
+minstrel_ht_fill_rate_array(u8 *dest, struct ieee80211_supported_band *sband,
+                           const s16 *bitrates, int n_rates, u32 rate_flags)
 {
-       static const int bitrates[4] = { 10, 20, 55, 110 };
-       struct ieee80211_supported_band *sband;
-       u32 rate_flags = ieee80211_chandef_rate_flags(&mp->hw->conf.chandef);
        int i, j;
 
-       sband = mp->hw->wiphy->bands[NL80211_BAND_2GHZ];
-       if (!sband)
-               return;
-
        for (i = 0; i < sband->n_bitrates; i++) {
                struct ieee80211_rate *rate = &sband->bitrates[i];
 
-               if (rate->flags & IEEE80211_RATE_ERP_G)
-                       continue;
-
                if ((rate_flags & sband->bitrates[i].flags) != rate_flags)
                        continue;
 
-               for (j = 0; j < ARRAY_SIZE(bitrates); j++) {
+               for (j = 0; j < n_rates; j++) {
                        if (rate->bitrate != bitrates[j])
                                continue;
 
-                       mp->cck_rates[j] = i;
+                       dest[j] = i;
                        break;
                }
        }
 }
 
+static void
+minstrel_ht_init_cck_rates(struct minstrel_priv *mp)
+{
+       static const s16 bitrates[4] = { 10, 20, 55, 110 };
+       struct ieee80211_supported_band *sband;
+       u32 rate_flags = ieee80211_chandef_rate_flags(&mp->hw->conf.chandef);
+
+       memset(mp->cck_rates, 0xff, sizeof(mp->cck_rates));
+       sband = mp->hw->wiphy->bands[NL80211_BAND_2GHZ];
+       if (!sband)
+               return;
+
+       BUILD_BUG_ON(ARRAY_SIZE(mp->cck_rates) != ARRAY_SIZE(bitrates));
+       minstrel_ht_fill_rate_array(mp->cck_rates, sband,
+                                   minstrel_cck_bitrates,
+                                   ARRAY_SIZE(minstrel_cck_bitrates),
+                                   rate_flags);
+}
+
+static void
+minstrel_ht_init_ofdm_rates(struct minstrel_priv *mp, enum nl80211_band band)
+{
+       static const s16 bitrates[8] = { 60, 90, 120, 180, 240, 360, 480, 540 };
+       struct ieee80211_supported_band *sband;
+       u32 rate_flags = ieee80211_chandef_rate_flags(&mp->hw->conf.chandef);
+
+       memset(mp->ofdm_rates[band], 0xff, sizeof(mp->ofdm_rates[band]));
+       sband = mp->hw->wiphy->bands[band];
+       if (!sband)
+               return;
+
+       BUILD_BUG_ON(ARRAY_SIZE(mp->ofdm_rates[band]) != ARRAY_SIZE(bitrates));
+       minstrel_ht_fill_rate_array(mp->ofdm_rates[band], sband,
+                                   minstrel_ofdm_bitrates,
+                                   ARRAY_SIZE(minstrel_ofdm_bitrates),
+                                   rate_flags);
+}
+
 static void *
 minstrel_ht_alloc(struct ieee80211_hw *hw)
 {
        struct minstrel_priv *mp;
+       int i;
 
        mp = kzalloc(sizeof(struct minstrel_priv), GFP_ATOMIC);
        if (!mp)
        mp->new_avg = true;
 
        minstrel_ht_init_cck_rates(mp);
+       for (i = 0; i < ARRAY_SIZE(mp->hw->wiphy->bands); i++)
+           minstrel_ht_init_ofdm_rates(mp, i);
 
        return mp;
 }
        struct minstrel_ht_sta *mi = &msp->ht;
        int i, j, prob, tp_avg;
 
-       if (!msp->is_ht)
-               return mac80211_minstrel.get_expected_throughput(priv_sta);
-
        i = mi->max_tp_rate[0] / MCS_GROUP_RATES;
        j = mi->max_tp_rate[0] % MCS_GROUP_RATES;
        prob = mi->groups[i].rates[j].prob_avg;