}
 }
 
-static void xe_pt_abort_bind(struct xe_vma *vma,
-                            struct xe_vm_pgtable_update *entries,
-                            u32 num_entries)
+static void xe_pt_cancel_bind(struct xe_vma *vma,
+                             struct xe_vm_pgtable_update *entries,
+                             u32 num_entries)
 {
        u32 i, j;
 
        for (i = 0; i < num_entries; i++) {
-               if (!entries[i].pt_entries)
+               struct xe_pt *pt = entries[i].pt;
+
+               if (!pt)
                        continue;
 
-               for (j = 0; j < entries[i].qwords; j++)
-                       xe_pt_destroy(entries[i].pt_entries[j].pt, xe_vma_vm(vma)->flags, NULL);
+               if (pt->level) {
+                       for (j = 0; j < entries[i].qwords; j++)
+                               xe_pt_destroy(entries[i].pt_entries[j].pt,
+                                             xe_vma_vm(vma)->flags, NULL);
+               }
+
                kfree(entries[i].pt_entries);
+               entries[i].pt_entries = NULL;
+               entries[i].qwords = 0;
        }
 }
 
        xe_vm_assert_held(vm);
 }
 
-static void xe_pt_commit_bind(struct xe_vma *vma,
-                             struct xe_vm_pgtable_update *entries,
-                             u32 num_entries, bool rebind,
-                             struct llist_head *deferred)
+static void xe_pt_commit(struct xe_vma *vma,
+                        struct xe_vm_pgtable_update *entries,
+                        u32 num_entries, struct llist_head *deferred)
+{
+       u32 i, j;
+
+       xe_pt_commit_locks_assert(vma);
+
+       for (i = 0; i < num_entries; i++) {
+               struct xe_pt *pt = entries[i].pt;
+
+               if (!pt->level)
+                       continue;
+
+               for (j = 0; j < entries[i].qwords; j++) {
+                       struct xe_pt *oldpte = entries[i].pt_entries[j].pt;
+
+                       xe_pt_destroy(oldpte, xe_vma_vm(vma)->flags, deferred);
+               }
+       }
+}
+
+static void xe_pt_abort_bind(struct xe_vma *vma,
+                            struct xe_vm_pgtable_update *entries,
+                            u32 num_entries, bool rebind)
+{
+       int i, j;
+
+       xe_pt_commit_locks_assert(vma);
+
+       for (i = num_entries - 1; i >= 0; --i) {
+               struct xe_pt *pt = entries[i].pt;
+               struct xe_pt_dir *pt_dir;
+
+               if (!rebind)
+                       pt->num_live -= entries[i].qwords;
+
+               if (!pt->level)
+                       continue;
+
+               pt_dir = as_xe_pt_dir(pt);
+               for (j = 0; j < entries[i].qwords; j++) {
+                       u32 j_ = j + entries[i].ofs;
+                       struct xe_pt *newpte = xe_pt_entry(pt_dir, j_);
+                       struct xe_pt *oldpte = entries[i].pt_entries[j].pt;
+
+                       pt_dir->children[j_] = oldpte ? &oldpte->base : 0;
+                       xe_pt_destroy(newpte, xe_vma_vm(vma)->flags, NULL);
+               }
+       }
+}
+
+static void xe_pt_commit_prepare_bind(struct xe_vma *vma,
+                                     struct xe_vm_pgtable_update *entries,
+                                     u32 num_entries, bool rebind)
 {
        u32 i, j;
 
                for (j = 0; j < entries[i].qwords; j++) {
                        u32 j_ = j + entries[i].ofs;
                        struct xe_pt *newpte = entries[i].pt_entries[j].pt;
+                       struct xe_pt *oldpte = NULL;
 
                        if (xe_pt_entry(pt_dir, j_))
-                               xe_pt_destroy(xe_pt_entry(pt_dir, j_),
-                                             xe_vma_vm(vma)->flags, deferred);
+                               oldpte = xe_pt_entry(pt_dir, j_);
 
                        pt_dir->children[j_] = &newpte->base;
+                       entries[i].pt_entries[j].pt = oldpte;
                }
        }
 }
        err = xe_pt_stage_bind(tile, vma, entries, num_entries);
        if (!err)
                xe_tile_assert(tile, *num_entries);
-       else /* abort! */
-               xe_pt_abort_bind(vma, entries, *num_entries);
 
        return err;
 }
                                     ifence->end, ifence->asid);
 }
 
-static int invalidation_fence_init(struct xe_gt *gt,
-                                  struct invalidation_fence *ifence,
-                                  struct dma_fence *fence,
-                                  u64 start, u64 end, u32 asid)
+static void invalidation_fence_init(struct xe_gt *gt,
+                                   struct invalidation_fence *ifence,
+                                   struct dma_fence *fence,
+                                   u64 start, u64 end, u32 asid)
 {
        int ret;
 
        }
 
        xe_gt_assert(gt, !ret || ret == -ENOENT);
-
-       return ret && ret != -ENOENT ? ret : 0;
 }
 
 struct xe_pt_stage_unbind_walk {
        struct xe_pt *xe_child = container_of(*child, typeof(*xe_child), base);
        pgoff_t end_offset;
        u64 size = 1ull << walk->shifts[--level];
+       int err;
 
        if (!IS_ALIGNED(addr, size))
                addr = xe_walk->modified_start;
                                     &end_offset))
                return 0;
 
-       (void)xe_pt_new_shared(&xe_walk->wupd, xe_child, offset, false);
+       err = xe_pt_new_shared(&xe_walk->wupd, xe_child, offset, true);
+       if (err)
+               return err;
+
        xe_walk->wupd.updates[level].update->qwords = end_offset - offset;
 
        return 0;
                memset64(ptr, empty, num_qwords);
 }
 
+static void xe_pt_abort_unbind(struct xe_vma *vma,
+                              struct xe_vm_pgtable_update *entries,
+                              u32 num_entries)
+{
+       int i, j;
+
+       xe_pt_commit_locks_assert(vma);
+
+       for (i = num_entries - 1; i >= 0; --i) {
+               struct xe_vm_pgtable_update *entry = &entries[i];
+               struct xe_pt *pt = entry->pt;
+               struct xe_pt_dir *pt_dir = as_xe_pt_dir(pt);
+
+               pt->num_live += entry->qwords;
+
+               if (!pt->level)
+                       continue;
+
+               for (j = entry->ofs; j < entry->ofs + entry->qwords; j++)
+                       pt_dir->children[i] =
+                               entries[i].pt_entries[j - entry->ofs].pt ?
+                               &entries[i].pt_entries[j - entry->ofs].pt->base : NULL;
+       }
+}
+
 static void
-xe_pt_commit_unbind(struct xe_vma *vma,
-                   struct xe_vm_pgtable_update *entries, u32 num_entries,
-                   struct llist_head *deferred)
+xe_pt_commit_prepare_unbind(struct xe_vma *vma,
+                           struct xe_vm_pgtable_update *entries,
+                           u32 num_entries)
 {
-       u32 j;
+       int i, j;
 
        xe_pt_commit_locks_assert(vma);
 
-       for (j = 0; j < num_entries; ++j) {
-               struct xe_vm_pgtable_update *entry = &entries[j];
+       for (i = 0; i < num_entries; ++i) {
+               struct xe_vm_pgtable_update *entry = &entries[i];
                struct xe_pt *pt = entry->pt;
+               struct xe_pt_dir *pt_dir;
 
                pt->num_live -= entry->qwords;
-               if (pt->level) {
-                       struct xe_pt_dir *pt_dir = as_xe_pt_dir(pt);
-                       u32 i;
-
-                       for (i = entry->ofs; i < entry->ofs + entry->qwords;
-                            i++) {
-                               if (xe_pt_entry(pt_dir, i))
-                                       xe_pt_destroy(xe_pt_entry(pt_dir, i),
-                                                     xe_vma_vm(vma)->flags, deferred);
+               if (!pt->level)
+                       continue;
 
-                               pt_dir->children[i] = NULL;
-                       }
+               pt_dir = as_xe_pt_dir(pt);
+               for (j = entry->ofs; j < entry->ofs + entry->qwords; j++) {
+                       entry->pt_entries[j - entry->ofs].pt =
+                               xe_pt_entry(pt_dir, j);
+                       pt_dir->children[j] = NULL;
                }
        }
 }
 {
        u32 current_op = pt_update_ops->current_op;
        struct xe_vm_pgtable_update_op *pt_op = &pt_update_ops->ops[current_op];
-       struct llist_head *deferred = &pt_update_ops->deferred;
        int err;
 
        xe_bo_assert_held(xe_vma_bo(vma));
                        /* We bump also if batch_invalidate_tlb is true */
                        vm->tlb_flush_seqno++;
 
-               /* FIXME: Don't commit right away */
                vma->tile_staged |= BIT(tile->id);
                pt_op->vma = vma;
-               xe_pt_commit_bind(vma, pt_op->entries, pt_op->num_entries,
-                                 pt_op->rebind, deferred);
+               xe_pt_commit_prepare_bind(vma, pt_op->entries,
+                                         pt_op->num_entries, pt_op->rebind);
+       } else {
+               xe_pt_cancel_bind(vma, pt_op->entries, pt_op->num_entries);
        }
 
        return err;
 {
        u32 current_op = pt_update_ops->current_op;
        struct xe_vm_pgtable_update_op *pt_op = &pt_update_ops->ops[current_op];
-       struct llist_head *deferred = &pt_update_ops->deferred;
        int err;
 
        if (!((vma->tile_present | vma->tile_staged) & BIT(tile->id)))
        pt_update_ops->needs_userptr_lock |= xe_vma_is_userptr(vma);
        pt_update_ops->needs_invalidation = true;
 
-       /* FIXME: Don't commit right away */
-       xe_pt_commit_unbind(vma, pt_op->entries, pt_op->num_entries,
-                           deferred);
+       xe_pt_commit_prepare_unbind(vma, pt_op->entries, pt_op->num_entries);
 
        return 0;
 }
        struct invalidation_fence *ifence = NULL;
        struct xe_range_fence *rfence;
        struct xe_vma_op *op;
-       int err = 0;
+       int err = 0, i;
        struct xe_migrate_pt_update update = {
                .ops = pt_update_ops->needs_userptr_lock ?
                        &userptr_migrate_ops :
 
        if (pt_update_ops->needs_invalidation) {
                ifence = kzalloc(sizeof(*ifence), GFP_KERNEL);
-               if (!ifence)
-                       return ERR_PTR(-ENOMEM);
+               if (!ifence) {
+                       err = -ENOMEM;
+                       goto kill_vm_tile1;
+               }
        }
 
        rfence = kzalloc(sizeof(*rfence), GFP_KERNEL);
                goto free_rfence;
        }
 
+       /* Point of no return - VM killed if failure after this */
+       for (i = 0; i < pt_update_ops->current_op; ++i) {
+               struct xe_vm_pgtable_update_op *pt_op = &pt_update_ops->ops[i];
+
+               xe_pt_commit(pt_op->vma, pt_op->entries,
+                            pt_op->num_entries, &pt_update_ops->deferred);
+               pt_op->vma = NULL;      /* skip in xe_pt_update_ops_abort */
+       }
+
        if (xe_range_fence_insert(&vm->rftree[tile->id], rfence,
                                  &xe_range_fence_kfree_ops,
                                  pt_update_ops->start,
 
        /* tlb invalidation must be done before signaling rebind */
        if (ifence) {
-               err = invalidation_fence_init(tile->primary_gt, ifence, fence,
-                                             pt_update_ops->start,
-                                             pt_update_ops->last,
-                                             vm->usm.asid);
-               if (err)
-                       goto put_fence;
+               invalidation_fence_init(tile->primary_gt, ifence, fence,
+                                       pt_update_ops->start,
+                                       pt_update_ops->last, vm->usm.asid);
                fence = &ifence->base.base;
        }
 
 
        return fence;
 
-put_fence:
-       if (pt_update_ops->needs_userptr_lock)
-               up_read(&vm->userptr.notifier_lock);
-       dma_fence_put(fence);
 free_rfence:
        kfree(rfence);
 free_ifence:
        kfree(ifence);
+kill_vm_tile1:
+       if (err != -EAGAIN && tile->id)
+               xe_vm_kill(vops->vm, false);
 
        return ERR_PTR(err);
 }
        lockdep_assert_held(&vops->vm->lock);
        xe_vm_assert_held(vops->vm);
 
-       /* FIXME: Not 100% correct */
-       for (i = 0; i < pt_update_ops->num_ops; ++i) {
+       for (i = 0; i < pt_update_ops->current_op; ++i) {
                struct xe_vm_pgtable_update_op *pt_op = &pt_update_ops->ops[i];
 
-               if (pt_op->bind)
-                       xe_pt_free_bind(pt_op->entries, pt_op->num_entries);
+               xe_pt_free_bind(pt_op->entries, pt_op->num_entries);
        }
        xe_bo_put_commit(&vops->pt_update_ops[tile->id].deferred);
 }
  */
 void xe_pt_update_ops_abort(struct xe_tile *tile, struct xe_vma_ops *vops)
 {
+       struct xe_vm_pgtable_update_ops *pt_update_ops =
+               &vops->pt_update_ops[tile->id];
+       int i;
+
        lockdep_assert_held(&vops->vm->lock);
        xe_vm_assert_held(vops->vm);
 
-       /* FIXME: Just kill VM for now + cleanup PTs */
+       for (i = pt_update_ops->num_ops - 1; i >= 0; --i) {
+               struct xe_vm_pgtable_update_op *pt_op =
+                       &pt_update_ops->ops[i];
+
+               if (!pt_op->vma || i >= pt_update_ops->current_op)
+                       continue;
+
+               if (pt_op->bind)
+                       xe_pt_abort_bind(pt_op->vma, pt_op->entries,
+                                        pt_op->num_entries,
+                                        pt_op->rebind);
+               else
+                       xe_pt_abort_unbind(pt_op->vma, pt_op->entries,
+                                          pt_op->num_entries);
+       }
+
        xe_bo_put_commit(&vops->pt_update_ops[tile->id].deferred);
-       xe_vm_kill(vops->vm, false);
 }