summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorOwain Ainsworth <oga@cvs.openbsd.org>2010-05-08 21:33:50 +0000
committerOwain Ainsworth <oga@cvs.openbsd.org>2010-05-08 21:33:50 +0000
commit6b7eae65182712ef543c8069d211991a47cbcda4 (patch)
treeaed68744ba53608309619649192c1a6e37732018
parent211a77908cfe760ff4df3dfc6e813e826b3fcc82 (diff)
A little bit of a hack.
If userland asks to allocate an object large enough that two that size could not fit around the pinned objects, disallow it with EFBIG. This prevents mmap of large objects that big and copying between them putting the machine into infinite thrashing. with a patch to the ddx (on my git branch) that allocates a non-accelerated pixmap when it gets that return code, matthieu@s test huge image works happily when before it DOSed the kernel. The correct fix would be to fall back to mmaping the backing pages for objects that big (radeondrm will need such ability anyway). This however is a lot more complicated and I am still working out how to do it correctly hence this commit for now.
-rw-r--r--sys/dev/pci/drm/i915_drv.c32
-rw-r--r--sys/dev/pci/drm/i915_drv.h1
2 files changed, 32 insertions, 1 deletions
diff --git a/sys/dev/pci/drm/i915_drv.c b/sys/dev/pci/drm/i915_drv.c
index b2469523d1e..db3b240b62c 100644
--- a/sys/dev/pci/drm/i915_drv.c
+++ b/sys/dev/pci/drm/i915_drv.c
@@ -79,6 +79,7 @@ int inteldrm_fault(struct drm_obj *, struct uvm_faultinfo *, off_t,
vaddr_t, vm_page_t *, int, int, vm_prot_t, int );
void inteldrm_wipe_mappings(struct drm_obj *);
void inteldrm_purge_obj(struct drm_obj *);
+void inteldrm_set_max_obj_size(struct drm_i915_private *);
/* For reset and suspend */
int inteldrm_save_state(struct drm_i915_private *);
@@ -976,12 +977,27 @@ i915_gem_init_ioctl(struct drm_device *dev, void *data,
}
dev->gtt_total = (uint32_t)(args->gtt_end - args->gtt_start);
+ inteldrm_set_max_obj_size(dev_priv);
DRM_UNLOCK();
return 0;
}
+void
+inteldrm_set_max_obj_size(struct drm_i915_private *dev_priv)
+{
+ struct drm_device *dev = (struct drm_device *)dev_priv->drmdev;
+
+ /*
+ * Allow max obj size up to the size where ony 2 would fit the
+ * aperture, but some slop exists due to alignment etc
+ */
+ dev_priv->max_gem_obj_size = (dev->gtt_total -
+ atomic_read(&dev->pin_memory)) * 3 / 4 / 2;
+
+}
+
int
i915_gem_get_aperture_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
@@ -1005,11 +1021,20 @@ int
i915_gem_create_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
+ struct drm_i915_private *dev_priv = dev->dev_private;
struct drm_i915_gem_create *args = data;
struct drm_obj *obj;
int handle, ret;
args->size = round_page(args->size);
+ /*
+ * XXX to avoid copying between 2 objs more than half the aperture size
+ * we don't allow allocations that are that big. This will be fixed
+ * eventually by intelligently falling back to cpu reads/writes in
+ * such cases. (linux allows this but does cpu maps in the ddx instead).
+ */
+ if (args->size > dev_priv->max_gem_obj_size)
+ return (EFBIG);
/* Allocate the new object */
obj = drm_gem_object_alloc(dev, args->size);
@@ -3453,6 +3478,7 @@ int
i915_gem_pin_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
+ struct drm_i915_private *dev_priv = dev->dev_private;
struct drm_i915_gem_pin *args = data;
struct drm_obj *obj;
struct inteldrm_obj *obj_priv;
@@ -3475,6 +3501,7 @@ i915_gem_pin_ioctl(struct drm_device *dev, void *data,
ret = i915_gem_object_pin(obj, args->alignment, 1);
if (ret != 0)
goto out;
+ inteldrm_set_max_obj_size(dev_priv);
}
/* XXX - flush the CPU caches for pinned objects
@@ -3494,6 +3521,7 @@ int
i915_gem_unpin_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
+ struct drm_i915_private *dev_priv = dev->dev_private;
struct drm_i915_gem_pin *args = data;
struct inteldrm_obj *obj_priv;
struct drm_obj *obj;
@@ -3512,8 +3540,10 @@ i915_gem_unpin_ioctl(struct drm_device *dev, void *data,
goto out;
}
- if (--obj_priv->user_pin_count == 0)
+ if (--obj_priv->user_pin_count == 0) {
i915_gem_object_unpin(obj);
+ inteldrm_set_max_obj_size(dev_priv);
+ }
out:
drm_unhold_and_unref(obj);
diff --git a/sys/dev/pci/drm/i915_drv.h b/sys/dev/pci/drm/i915_drv.h
index ba62e0906a0..45a24a0e168 100644
--- a/sys/dev/pci/drm/i915_drv.h
+++ b/sys/dev/pci/drm/i915_drv.h
@@ -124,6 +124,7 @@ typedef struct drm_i915_private {
void *hw_status_page;
unsigned int status_gfx_addr;
u_int32_t counter;
+ size_t max_gem_obj_size; /* XXX */
/* Protects user_irq_refcount and irq_mask reg */
struct mutex user_irq_lock;