return vsi;
 }
 
-/**
- * ice_calc_vf_first_vector_idx - Calculate MSIX vector index in the PF space
- * @pf: pointer to PF structure
- * @vf: pointer to VF that the first MSIX vector index is being calculated for
- *
- * This returns the first MSIX vector index in PF space that is used by this VF.
- * This index is used when accessing PF relative registers such as
- * GLINT_VECT2FUNC and GLINT_DYN_CTL.
- * This will always be the OICR index in the AVF driver so any functionality
- * using vf->first_vector_idx for queue configuration will have to increment by
- * 1 to avoid meddling with the OICR index.
- */
-static int ice_calc_vf_first_vector_idx(struct ice_pf *pf, struct ice_vf *vf)
-{
-       return pf->sriov_base_vector + vf->vf_id * pf->vfs.num_msix_per;
-}
 
 /**
  * ice_ena_vf_msix_mappings - enable VF MSIX mappings in hardware
        return 0;
 }
 
+/**
+ * ice_sriov_get_irqs - get irqs for SR-IOV usacase
+ * @pf: pointer to PF structure
+ * @needed: number of irqs to get
+ *
+ * This returns the first MSI-X vector index in PF space that is used by this
+ * VF. This index is used when accessing PF relative registers such as
+ * GLINT_VECT2FUNC and GLINT_DYN_CTL.
+ * This will always be the OICR index in the AVF driver so any functionality
+ * using vf->first_vector_idx for queue configuration_id: id of VF which will
+ * use this irqs
+ *
+ * Only SRIOV specific vectors are tracked in sriov_irq_bm. SRIOV vectors are
+ * allocated from the end of global irq index. First bit in sriov_irq_bm means
+ * last irq index etc. It simplifies extension of SRIOV vectors.
+ * They will be always located from sriov_base_vector to the last irq
+ * index. While increasing/decreasing sriov_base_vector can be moved.
+ */
+static int ice_sriov_get_irqs(struct ice_pf *pf, u16 needed)
+{
+       int res = bitmap_find_next_zero_area(pf->sriov_irq_bm,
+                                            pf->sriov_irq_size, 0, needed, 0);
+       /* conversion from number in bitmap to global irq index */
+       int index = pf->sriov_irq_size - res - needed;
+
+       if (res >= pf->sriov_irq_size || index < pf->sriov_base_vector)
+               return -ENOENT;
+
+       bitmap_set(pf->sriov_irq_bm, res, needed);
+       return index;
+}
+
+/**
+ * ice_sriov_free_irqs - free irqs used by the VF
+ * @pf: pointer to PF structure
+ * @vf: pointer to VF structure
+ */
+static void ice_sriov_free_irqs(struct ice_pf *pf, struct ice_vf *vf)
+{
+       /* Move back from first vector index to first index in bitmap */
+       int bm_i = pf->sriov_irq_size - vf->first_vector_idx - vf->num_msix;
+
+       bitmap_clear(pf->sriov_irq_bm, bm_i, vf->num_msix);
+       vf->first_vector_idx = 0;
+}
+
 /**
  * ice_init_vf_vsi_res - initialize/setup VF VSI resources
  * @vf: VF to initialize/setup the VSI for
        struct ice_vsi *vsi;
        int err;
 
-       vf->first_vector_idx = ice_calc_vf_first_vector_idx(pf, vf);
+       vf->first_vector_idx = ice_sriov_get_irqs(pf, vf->num_msix);
+       if (vf->first_vector_idx < 0)
+               return -ENOMEM;
 
        vsi = ice_vf_vsi_setup(vf);
        if (!vsi)
        return pf->sriov_irq_size - ice_get_max_used_msix_vector(pf);
 }
 
+static int ice_sriov_move_base_vector(struct ice_pf *pf, int move)
+{
+       if (pf->sriov_base_vector - move < ice_get_max_used_msix_vector(pf))
+               return -ENOMEM;
+
+       pf->sriov_base_vector -= move;
+       return 0;
+}
+
+static void ice_sriov_remap_vectors(struct ice_pf *pf, u16 restricted_id)
+{
+       u16 vf_ids[ICE_MAX_SRIOV_VFS];
+       struct ice_vf *tmp_vf;
+       int to_remap = 0, bkt;
+
+       /* For better irqs usage try to remap irqs of VFs
+        * that aren't running yet
+        */
+       ice_for_each_vf(pf, bkt, tmp_vf) {
+               /* skip VF which is changing the number of MSI-X */
+               if (restricted_id == tmp_vf->vf_id ||
+                   test_bit(ICE_VF_STATE_ACTIVE, tmp_vf->vf_states))
+                       continue;
+
+               ice_dis_vf_mappings(tmp_vf);
+               ice_sriov_free_irqs(pf, tmp_vf);
+
+               vf_ids[to_remap] = tmp_vf->vf_id;
+               to_remap += 1;
+       }
+
+       for (int i = 0; i < to_remap; i++) {
+               tmp_vf = ice_get_vf_by_id(pf, vf_ids[i]);
+               if (!tmp_vf)
+                       continue;
+
+               tmp_vf->first_vector_idx =
+                       ice_sriov_get_irqs(pf, tmp_vf->num_msix);
+               /* there is no need to rebuild VSI as we are only changing the
+                * vector indexes not amount of MSI-X or queues
+                */
+               ice_ena_vf_mappings(tmp_vf);
+               ice_put_vf(tmp_vf);
+       }
+}
+
 /**
  * ice_sriov_set_msix_vec_count
  * @vf_dev: pointer to pci_dev struct of VF device
 {
        struct pci_dev *pdev = pci_physfn(vf_dev);
        struct ice_pf *pf = pci_get_drvdata(pdev);
+       u16 prev_msix, prev_queues, queues;
+       bool needs_rebuild = false;
        struct ice_vf *vf;
-       u16 queues;
        int id;
 
        if (!ice_get_num_vfs(pf))
        /* add 1 MSI-X for OICR */
        msix_vec_count += 1;
 
+       if (queues > min(ice_get_avail_txq_count(pf),
+                        ice_get_avail_rxq_count(pf)))
+               return -EINVAL;
+
+       if (msix_vec_count < ICE_MIN_INTR_PER_VF)
+               return -EINVAL;
+
        /* Transition of PCI VF function number to function_id */
        for (id = 0; id < pci_num_vf(pdev); id++) {
                if (vf_dev->devfn == pci_iov_virtfn_devfn(pdev, id))
        if (!vf)
                return -ENOENT;
 
+       prev_msix = vf->num_msix;
+       prev_queues = vf->num_vf_qs;
+
+       if (ice_sriov_move_base_vector(pf, msix_vec_count - prev_msix)) {
+               ice_put_vf(vf);
+               return -ENOSPC;
+       }
+
        ice_dis_vf_mappings(vf);
+       ice_sriov_free_irqs(pf, vf);
+
+       /* Remap all VFs beside the one is now configured */
+       ice_sriov_remap_vectors(pf, vf->vf_id);
+
        vf->num_msix = msix_vec_count;
        vf->num_vf_qs = queues;
-       ice_vsi_rebuild(ice_get_vf_vsi(vf), ICE_VSI_FLAG_NO_INIT);
+       vf->first_vector_idx = ice_sriov_get_irqs(pf, vf->num_msix);
+       if (vf->first_vector_idx < 0)
+               goto unroll;
+
+       ice_vf_vsi_release(vf);
+       if (vf->vf_ops->create_vsi(vf)) {
+               /* Try to rebuild with previous values */
+               needs_rebuild = true;
+               goto unroll;
+       }
+
+       dev_info(ice_pf_to_dev(pf),
+                "Changing VF %d resources to %d vectors and %d queues\n",
+                vf->vf_id, vf->num_msix, vf->num_vf_qs);
+
        ice_ena_vf_mappings(vf);
        ice_put_vf(vf);
 
        return 0;
+
+unroll:
+       dev_info(ice_pf_to_dev(pf),
+                "Can't set %d vectors on VF %d, falling back to %d\n",
+                vf->num_msix, vf->vf_id, prev_msix);
+
+       vf->num_msix = prev_msix;
+       vf->num_vf_qs = prev_queues;
+       vf->first_vector_idx = ice_sriov_get_irqs(pf, vf->num_msix);
+       if (vf->first_vector_idx < 0)
+               return -EINVAL;
+
+       if (needs_rebuild)
+               vf->vf_ops->create_vsi(vf);
+
+       ice_ena_vf_mappings(vf);
+       ice_put_vf(vf);
+
+       return -EINVAL;
 }
 
 /**