#define DEFAULT_POLL_PERIOD_NS (NSEC_PER_SEC / DEFAULT_POLL_FREQUENCY_HZ)
 #define XE_OA_UNIT_INVALID U32_MAX
 
+enum xe_oam_unit_type {
+       XE_OAM_UNIT_SAG,
+       XE_OAM_UNIT_SCMI_0,
+       XE_OAM_UNIT_SCMI_1,
+};
+
 enum xe_oa_submit_deps {
        XE_OA_SUBMIT_NO_DEPS,
        XE_OA_SUBMIT_ADD_DEPS,
                return type == DRM_XE_OA_FMT_TYPE_OAG || type == DRM_XE_OA_FMT_TYPE_OAR ||
                        type == DRM_XE_OA_FMT_TYPE_OAC || type == DRM_XE_OA_FMT_TYPE_PEC;
        case DRM_XE_OA_UNIT_TYPE_OAM:
+       case DRM_XE_OA_UNIT_TYPE_OAM_SAG:
                return type == DRM_XE_OA_FMT_TYPE_OAM || type == DRM_XE_OA_FMT_TYPE_OAM_MPEC;
        default:
                return false;
 
 static u32 num_oa_units_per_gt(struct xe_gt *gt)
 {
-       return 1;
+       if (!xe_gt_is_media_type(gt) || GRAPHICS_VER(gt_to_xe(gt)) < 20)
+               return 1;
+       else if (!IS_DGFX(gt_to_xe(gt)))
+               return XE_OAM_UNIT_SCMI_0 + 1; /* SAG + SCMI_0 */
+       else
+               return XE_OAM_UNIT_SCMI_1 + 1; /* SAG + SCMI_0 + SCMI_1 */
 }
 
 static u32 __hwe_oam_unit(struct xe_hw_engine *hwe)
 {
-       if (GRAPHICS_VERx100(gt_to_xe(hwe->gt)) >= 1270) {
-               /*
-                * There's 1 SAMEDIA gt and 1 OAM per SAMEDIA gt. All media slices
-                * within the gt use the same OAM. All MTL/LNL SKUs list 1 SA MEDIA
-                */
-               xe_gt_WARN_ON(hwe->gt, hwe->gt->info.type != XE_GT_TYPE_MEDIA);
+       if (GRAPHICS_VERx100(gt_to_xe(hwe->gt)) < 1270)
+               return XE_OA_UNIT_INVALID;
 
+       xe_gt_WARN_ON(hwe->gt, !xe_gt_is_media_type(hwe->gt));
+
+       if (GRAPHICS_VER(gt_to_xe(hwe->gt)) < 20)
                return 0;
-       }
+       /*
+        * XE_OAM_UNIT_SAG has only GSCCS attached to it, but only on some platforms. Also
+        * GSCCS cannot be used to submit batches to program the OAM unit. Therefore we don't
+        * assign an OA unit to GSCCS. This means that XE_OAM_UNIT_SAG is exposed as an OA
+        * unit without attached engines. Fused off engines can also result in oa_unit's with
+        * num_engines == 0. OA streams can be opened on all OA units.
+        */
+       else if (hwe->engine_id == XE_HW_ENGINE_GSCCS0)
+               return XE_OA_UNIT_INVALID;
+       else if (!IS_DGFX(gt_to_xe(hwe->gt)))
+               return XE_OAM_UNIT_SCMI_0;
+       else if (hwe->class == XE_ENGINE_CLASS_VIDEO_DECODE)
+               return (hwe->instance / 2 & 0x1) + 1;
+       else if (hwe->class == XE_ENGINE_CLASS_VIDEO_ENHANCE)
+               return (hwe->instance & 0x1) + 1;
 
        return XE_OA_UNIT_INVALID;
 }
 
        case XE_ENGINE_CLASS_VIDEO_DECODE:
        case XE_ENGINE_CLASS_VIDEO_ENHANCE:
+       case XE_ENGINE_CLASS_OTHER:
                return __hwe_oam_unit(hwe);
 
        default:
 
 static void __xe_oa_init_oa_units(struct xe_gt *gt)
 {
-       const u32 mtl_oa_base[] = { 0x13000 };
+       /* Actual address is MEDIA_GT_GSI_OFFSET + oam_base_addr[i] */
+       const u32 oam_base_addr[] = {
+               [XE_OAM_UNIT_SAG]    = 0x13000,
+               [XE_OAM_UNIT_SCMI_0] = 0x14000,
+               [XE_OAM_UNIT_SCMI_1] = 0x14800,
+       };
        int i, num_units = gt->oa.num_oa_units;
 
        for (i = 0; i < num_units; i++) {
                struct xe_oa_unit *u = >->oa.oa_unit[i];
 
-               if (gt->info.type != XE_GT_TYPE_MEDIA) {
+               if (!xe_gt_is_media_type(gt)) {
                        u->regs = __oag_regs();
                        u->type = DRM_XE_OA_UNIT_TYPE_OAG;
-               } else if (GRAPHICS_VERx100(gt_to_xe(gt)) >= 1270) {
-                       u->regs = __oam_regs(mtl_oa_base[i]);
-                       u->type = DRM_XE_OA_UNIT_TYPE_OAM;
+               } else {
+                       xe_gt_assert(gt, GRAPHICS_VERx100(gt_to_xe(gt)) >= 1270);
+                       u->regs = __oam_regs(oam_base_addr[i]);
+                       u->type = i == XE_OAM_UNIT_SAG && GRAPHICS_VER(gt_to_xe(gt)) >= 20 ?
+                               DRM_XE_OA_UNIT_TYPE_OAM_SAG : DRM_XE_OA_UNIT_TYPE_OAM;
                }
 
                xe_mmio_write32(>->mmio, u->regs.oa_ctrl, 0);
                }
        }
 
-       /*
-        * Fused off engines can result in oa_unit's with num_engines == 0. These units
-        * will appear in OA unit query, but no OA streams can be opened on them.
-        */
        gt->oa.num_oa_units = num_oa_units;
        gt->oa.oa_unit = u;
 
        struct xe_gt *gt;
        int i, ret;
 
+       /* Needed for OAM implementation here */
+       BUILD_BUG_ON(XE_OAM_UNIT_SAG != 0);
+       BUILD_BUG_ON(XE_OAM_UNIT_SCMI_0 != 1);
+       BUILD_BUG_ON(XE_OAM_UNIT_SCMI_1 != 2);
+
        for_each_gt(gt, oa->xe, i) {
                ret = xe_oa_init_gt(gt);
                if (ret)