In some older devices, the min/max firmware API supported by
the driver depends on the specific device, when sharing the
the same MAC (config). For most newer devices, it really is
dependent on the MAC instead, since the firmware was frozen
for certain MAC types. However, in the future we expect also
freezes for RF types there.
To handle this most generally, add an API min/max to the MAC
config and then use the narrowest range prescribed by both,
if set.
For the newer MACs since 9000, move the configuration, there
was only a freeze on MAC+RF lines so far.
Signed-off-by: Johannes Berg <johannes.berg@intel.com>
Link: https://patch.msgid.link/20250509104454.2582160-2-miriam.rachel.korenblit@intel.com
Signed-off-by: Miri Korenblit <miriam.rachel.korenblit@intel.com>
                        .mask = 0xffffffff,
                },
        },
+       .ucode_api_min = IWL_22000_UCODE_API_MIN,
+       .ucode_api_max = IWL_22000_UCODE_API_MAX,
 };
 
 #define IWL_DEVICE_22500                                               \
-       .ucode_api_min = IWL_22000_UCODE_API_MIN,                       \
-       .ucode_api_max = IWL_22000_UCODE_API_MAX,                       \
        .led_mode = IWL_LED_RF_STATE,                                   \
        .non_shared_ant = ANT_B,                                        \
        .vht_mu_mimo_supported = true,                                  \
 
                        .mask = 0xffffffff,
                },
        },
+       .ucode_api_max = IWL9000_UCODE_API_MAX,
+       .ucode_api_min = IWL9000_UCODE_API_MIN,
 };
 
 static const struct iwl_tt_params iwl9000_tt_params = {
 };
 
 #define IWL_DEVICE_9000                                                        \
-       .ucode_api_max = IWL9000_UCODE_API_MAX,                         \
-       .ucode_api_min = IWL9000_UCODE_API_MIN,                         \
        .led_mode = IWL_LED_RF_STATE,                                   \
        .non_shared_ant = ANT_B,                                        \
        .dccm_offset = IWL9000_DCCM_OFFSET,                             \
 
                        .mask = DBGC_CUR_DBGBUF_STATUS_IDX_MSK,
                },
        },
+       .ucode_api_min = IWL_AX210_UCODE_API_MIN,
+       .ucode_api_max = IWL_AX210_UCODE_API_MAX,
 };
 
 #define IWL_DEVICE_AX210                                               \
-       .ucode_api_min = IWL_AX210_UCODE_API_MIN,                       \
-       .ucode_api_max = IWL_AX210_UCODE_API_MAX,                       \
        .led_mode = IWL_LED_RF_STATE,                                   \
        .non_shared_ant = ANT_B,                                        \
        .vht_mu_mimo_supported = true,                                  \
 
                },
        },
        .features = IWL_TX_CSUM_NETIF_FLAGS | NETIF_F_RXCSUM,
+       .ucode_api_max = IWL_BZ_UCODE_API_MAX,
+       .ucode_api_min = IWL_BZ_UCODE_API_MIN,
 };
 
 #define IWL_DEVICE_BZ                                                  \
-       .ucode_api_max = IWL_BZ_UCODE_API_MAX,                          \
-       .ucode_api_min = IWL_BZ_UCODE_API_MIN,                          \
        .ht_params = {                                                  \
                .stbc = true,                                           \
                .ldpc = true,                                           \
 
                },
        },
        .features = IWL_TX_CSUM_NETIF_FLAGS | NETIF_F_RXCSUM,
+       .ucode_api_max = IWL_DR_UCODE_API_MAX,
+       .ucode_api_min = IWL_DR_UCODE_API_MIN,
 };
 
 #define IWL_DEVICE_DR                                                  \
-       .ucode_api_max = IWL_DR_UCODE_API_MAX,                          \
-       .ucode_api_min = IWL_DR_UCODE_API_MIN,                          \
        .led_mode = IWL_LED_RF_STATE,                                   \
        .non_shared_ant = ANT_B,                                        \
        .vht_mu_mimo_supported = true,                                  \
 
                },
        },
        .features = IWL_TX_CSUM_NETIF_FLAGS | NETIF_F_RXCSUM,
+       .ucode_api_max = IWL_SC_UCODE_API_MAX,
+       .ucode_api_min = IWL_SC_UCODE_API_MIN,
 };
 
 #define IWL_DEVICE_SC                                                  \
-       .ucode_api_max = IWL_SC_UCODE_API_MAX,                          \
-       .ucode_api_min = IWL_SC_UCODE_API_MIN,                          \
        .led_mode = IWL_LED_RF_STATE,                                   \
        .non_shared_ant = ANT_B,                                        \
        .vht_mu_mimo_supported = true,                                  \
 
  * @mon_dbgi_regs: monitor DBGI registers
  * @mon_dram_regs: monitor DRAM registers
  * @mon_smem_regs: monitor SMEM registers
+ * @ucode_api_max: Highest version of uCode API supported by driver.
+ * @ucode_api_min: Lowest version of uCode API supported by driver.
  */
 struct iwl_family_base_params {
        unsigned int wd_timeout;
 
        u8 max_ll_items;
        u8 led_compensation;
+       u8 ucode_api_max;
+       u8 ucode_api_min;
        u32 mac_addr_from_csr:10;
        u8 nvm_hw_section_num;
        netdev_features_t features;
 
 static void iwl_req_fw_callback(const struct firmware *ucode_raw,
                                void *context);
 
+static void iwl_get_ucode_api_versions(struct iwl_trans *trans,
+                                      unsigned int *api_min,
+                                      unsigned int *api_max)
+{
+       const struct iwl_family_base_params *base = trans->mac_cfg->base;
+       const struct iwl_cfg *cfg = trans->cfg;
+
+       if (!base->ucode_api_max) {
+               *api_min = cfg->ucode_api_min;
+               *api_max = cfg->ucode_api_max;
+               return;
+       }
+
+       if (!cfg->ucode_api_max) {
+               *api_min = base->ucode_api_min;
+               *api_max = base->ucode_api_max;
+               return;
+       }
+
+       *api_min = max(cfg->ucode_api_min, base->ucode_api_min);
+       *api_max = min(cfg->ucode_api_max, base->ucode_api_max);
+}
+
 static int iwl_request_firmware(struct iwl_drv *drv, bool first)
 {
-       const struct iwl_cfg *cfg = drv->trans->cfg;
        char _fw_name_pre[FW_NAME_PRE_BUFSIZE];
+       unsigned int ucode_api_max, ucode_api_min;
        const char *fw_name_pre;
 
+       iwl_get_ucode_api_versions(drv->trans, &ucode_api_min, &ucode_api_max);
+
        if (drv->trans->mac_cfg->device_family == IWL_DEVICE_FAMILY_9000 &&
            (drv->trans->info.hw_rev_step != SILICON_B_STEP &&
             drv->trans->info.hw_rev_step != SILICON_C_STEP)) {
        fw_name_pre = iwl_drv_get_fwname_pre(drv->trans, _fw_name_pre);
 
        if (first)
-               drv->fw_index = cfg->ucode_api_max;
+               drv->fw_index = ucode_api_max;
        else
                drv->fw_index--;
 
-       if (drv->fw_index < cfg->ucode_api_min) {
+       if (drv->fw_index < ucode_api_min) {
                IWL_ERR(drv, "no suitable firmware found!\n");
 
-               if (cfg->ucode_api_min == cfg->ucode_api_max) {
+               if (ucode_api_min == ucode_api_max) {
                        IWL_ERR(drv, "%s-%d is required\n", fw_name_pre,
-                               cfg->ucode_api_max);
+                               ucode_api_max);
                } else {
                        IWL_ERR(drv, "minimum version required: %s-%d\n",
-                               fw_name_pre, cfg->ucode_api_min);
+                               fw_name_pre, ucode_api_min);
                        IWL_ERR(drv, "maximum version supported: %s-%d\n",
-                               fw_name_pre, cfg->ucode_api_max);
+                               fw_name_pre, ucode_api_max);
                }
 
                IWL_ERR(drv,
        struct iwlwifi_opmode_table *op;
        int err;
        struct iwl_firmware_pieces *pieces;
-       const unsigned int api_max = drv->trans->cfg->ucode_api_max;
-       const unsigned int api_min = drv->trans->cfg->ucode_api_min;
+       unsigned int api_min, api_max;
        size_t trigger_tlv_sz[FW_DBG_TRIGGER_MAX];
        u32 api_ver;
        int i;
        bool usniffer_images = false;
        bool failure = true;
 
+       iwl_get_ucode_api_versions(drv->trans, &api_min, &api_max);
+
        fw->ucode_capa.max_probe_length = IWL_DEFAULT_MAX_PROBE_LENGTH;
        fw->ucode_capa.standard_phy_calibration_size =
                        IWL_DEFAULT_STANDARD_PHY_CALIBRATE_TBL_SIZE;