#include <drm/drm_atomic_uapi.h>
 #include <drm/drm_atomic_helper.h>
 #include <drm/drm_blend.h>
+#include <drm/drm_fixed.h>
 #include <drm/drm_fourcc.h>
 #include <drm/drm_edid.h>
 #include <drm/drm_vblank.h>
        if (IS_ERR(mst_state))
                return PTR_ERR(mst_state);
 
-       if (!mst_state->pbn_div)
-               mst_state->pbn_div = dm_mst_get_pbn_divider(aconnector->mst_root->dc_link);
+       if (!mst_state->pbn_div.full)
+               mst_state->pbn_div.full = dfixed_const(dm_mst_get_pbn_divider(aconnector->mst_root->dc_link));
 
        if (!state->duplicated) {
                int max_bpc = conn_state->max_requested_bpc;
 
 #include <drm/drm_probe_helper.h>
 #include <drm/amdgpu_drm.h>
 #include <drm/drm_edid.h>
+#include <drm/drm_fixed.h>
 
 #include "dm_services.h"
 #include "amdgpu.h"
 
 static void dm_helpers_construct_old_payload(
                        struct dc_link *link,
-                       int pbn_per_slot,
+                       fixed20_12 pbn_per_slot_fp,
                        struct drm_dp_mst_atomic_payload *new_payload,
                        struct drm_dp_mst_atomic_payload *old_payload)
 {
        struct link_mst_stream_allocation_table current_link_table =
                                                                        link->mst_stream_alloc_table;
        struct link_mst_stream_allocation *dc_alloc;
+       int pbn_per_slot = dfixed_trunc(pbn_per_slot_fp);
        int i;
 
        *old_payload = *new_payload;
 
 #include <drm/display/drm_dp_mst_helper.h>
 #include <drm/drm_atomic.h>
 #include <drm/drm_atomic_helper.h>
+#include <drm/drm_fixed.h>
 #include "dm_services.h"
 #include "amdgpu.h"
 #include "amdgpu_dm.h"
                link_timeslots_used = 0;
 
                for (i = 0; i < count; i++)
-                       link_timeslots_used += DIV_ROUND_UP(vars[i + k].pbn, mst_state->pbn_div);
+                       link_timeslots_used += DIV_ROUND_UP(vars[i + k].pbn, dfixed_trunc(mst_state->pbn_div));
 
                fair_pbn_alloc =
-                       (63 - link_timeslots_used) / remaining_to_increase * mst_state->pbn_div;
+                       (63 - link_timeslots_used) / remaining_to_increase * dfixed_trunc(mst_state->pbn_div);
 
                if (initial_slack[next_index] > fair_pbn_alloc) {
                        vars[next_index].pbn += fair_pbn_alloc;
 
 #include <drm/drm_atomic_helper.h>
 #include <drm/drm_drv.h>
 #include <drm/drm_edid.h>
+#include <drm/drm_fixed.h>
 #include <drm/drm_print.h>
 #include <drm/drm_probe_helper.h>
 
  * value is in units of PBNs/(timeslots/1 MTP). This value can be used to
  * convert the number of PBNs required for a given stream to the number of
  * timeslots this stream requires in each MTP.
+ *
+ * Returns the BW / timeslot value in 20.12 fixed point format.
  */
-int drm_dp_get_vc_payload_bw(const struct drm_dp_mst_topology_mgr *mgr,
-                            int link_rate, int link_lane_count)
+fixed20_12 drm_dp_get_vc_payload_bw(const struct drm_dp_mst_topology_mgr *mgr,
+                                   int link_rate, int link_lane_count)
 {
+       fixed20_12 ret;
+
        if (link_rate == 0 || link_lane_count == 0)
                drm_dbg_kms(mgr->dev, "invalid link rate/lane count: (%d / %d)\n",
                            link_rate, link_lane_count);
 
        /* See DP v2.0 2.6.4.2, VCPayload_Bandwidth_for_OneTimeSlotPer_MTP_Allocation */
-       return link_rate * link_lane_count / 54000;
+       ret.full = dfixed_const(link_rate * link_lane_count / 54000);
+
+       return ret;
 }
 EXPORT_SYMBOL(drm_dp_get_vc_payload_bw);
 
                }
        }
 
-       req_slots = DIV_ROUND_UP(pbn, topology_state->pbn_div);
+       req_slots = DIV_ROUND_UP(pbn, dfixed_trunc(topology_state->pbn_div));
 
        drm_dbg_atomic(mgr->dev, "[CONNECTOR:%d:%s] [MST PORT:%p] TU %d -> %d\n",
                       port->connector->base.id, port->connector->name,
        state = to_drm_dp_mst_topology_state(mgr->base.state);
        seq_printf(m, "\n*** Atomic state info ***\n");
        seq_printf(m, "payload_mask: %x, max_payloads: %d, start_slot: %u, pbn_div: %d\n",
-                  state->payload_mask, mgr->max_payloads, state->start_slot, state->pbn_div);
+                  state->payload_mask, mgr->max_payloads, state->start_slot,
+                  dfixed_trunc(state->pbn_div));
 
        seq_printf(m, "\n| idx | port | vcpi | slots | pbn | dsc | status |     sink name     |\n");
        for (i = 0; i < mgr->max_payloads; i++) {
        }
 
        if (!payload_count)
-               mst_state->pbn_div = 0;
+               mst_state->pbn_div.full = dfixed_const(0);
 
        drm_dbg_atomic(mgr->dev, "[MST MGR:%p] mst state %p TU pbn_div=%d avail=%d used=%d\n",
-                      mgr, mst_state, mst_state->pbn_div, avail_slots,
+                      mgr, mst_state, dfixed_trunc(mst_state->pbn_div), avail_slots,
                       mst_state->total_avail_slots - avail_slots);
 
        return 0;
 
 #include <drm/drm_atomic.h>
 #include <drm/drm_atomic_helper.h>
 #include <drm/drm_edid.h>
+#include <drm/drm_fixed.h>
 #include <drm/drm_probe_helper.h>
 
 #include "i915_drv.h"
                 */
                drm_WARN_ON(&i915->drm, remote_m_n.tu < crtc_state->dp_m_n.tu);
                crtc_state->dp_m_n.tu = remote_m_n.tu;
-               crtc_state->pbn = remote_m_n.tu * mst_state->pbn_div;
+               crtc_state->pbn = remote_m_n.tu * dfixed_trunc(mst_state->pbn_div);
 
                slots = drm_dp_atomic_find_time_slots(state, &intel_dp->mst_mgr,
                                                      connector->port,
 
 #include <drm/drm_atomic_helper.h>
 #include <drm/drm_edid.h>
 #include <drm/drm_fb_helper.h>
+#include <drm/drm_fixed.h>
 #include <drm/drm_probe_helper.h>
 #include <drm/drm_vblank.h>
 
        if (ret == 0) {
                nvif_outp_dp_mst_vcpi(&mstm->outp->outp, msto->head->base.index,
                                      payload->vc_start_slot, payload->time_slots,
-                                     payload->pbn, payload->time_slots * mst_state->pbn_div);
+                                     payload->pbn,
+                                     payload->time_slots * dfixed_trunc(mst_state->pbn_div));
        } else {
                nvif_outp_dp_mst_vcpi(&mstm->outp->outp, msto->head->base.index, 0, 0, 0, 0);
        }
        if (IS_ERR(mst_state))
                return PTR_ERR(mst_state);
 
-       if (!mst_state->pbn_div) {
+       if (!mst_state->pbn_div.full) {
                struct nouveau_encoder *outp = mstc->mstm->outp;
 
                mst_state->pbn_div = drm_dp_get_vc_payload_bw(&mstm->mgr,
 
 #include <linux/types.h>
 #include <drm/display/drm_dp_helper.h>
 #include <drm/drm_atomic.h>
+#include <drm/drm_fixed.h>
 
 #if IS_ENABLED(CONFIG_DRM_DEBUG_DP_MST_TOPOLOGY_REFS)
 #include <linux/stackdepot.h>
         * @pbn_div: The current PBN divisor for this topology. The driver is expected to fill this
         * out itself.
         */
-       int pbn_div;
+       fixed20_12 pbn_div;
 };
 
 #define to_dp_mst_topology_mgr(x) container_of(x, struct drm_dp_mst_topology_mgr, base)
                                 struct drm_dp_mst_topology_mgr *mgr,
                                 struct drm_dp_mst_port *port);
 
-int drm_dp_get_vc_payload_bw(const struct drm_dp_mst_topology_mgr *mgr,
-                            int link_rate, int link_lane_count);
+fixed20_12 drm_dp_get_vc_payload_bw(const struct drm_dp_mst_topology_mgr *mgr,
+                                   int link_rate, int link_lane_count);
 
 int drm_dp_calc_pbn_mode(int clock, int bpp);