Commit 5572a004 authored by Zbigniew Kempczyński's avatar Zbigniew Kempczyński Committed by Rodrigo Vivi

drm/xe: Use nanoseconds instead of jiffies in uapi for user fence

Using jiffies as a timeout from userspace is weird even if
theoretically exists possiblity of acquiring jiffies via getconf.
Unfortunately this method is unreliable and the returned
value may vary from the one configured in the kernel config.

Now timeout is expressed in nanoseconds and its interpretation depends
on setting DRM_XE_UFENCE_WAIT_ABSTIME flag. Relative timeout (flag
is not set) means fence expire at now() + timeout. Absolute timeout
(flag is set) means that the fence expires at exact point of time.
Passing negative timeout means we will wait "forever" by setting
wait time to MAX_SCHEDULE_TIMEOUT.

Cc: Andi Shyti <andi.shyti@linux.intel.com>
Reviewed-by: default avatarAndi Shyti <andi.shyti@linux.intel.com>
Link: https://lore.kernel.org/r/20230628055141.398036-2-zbigniew.kempczynski@intel.comSigned-off-by: default avatarZbigniew Kempczyński <zbigniew.kempczynski@intel.com>
Signed-off-by: default avatarRodrigo Vivi <rodrigo.vivi@intel.com>
parent 2e60442a
...@@ -7,6 +7,7 @@ ...@@ -7,6 +7,7 @@
#include <drm/drm_device.h> #include <drm/drm_device.h>
#include <drm/drm_file.h> #include <drm/drm_file.h>
#include <drm/drm_utils.h>
#include <drm/xe_drm.h> #include <drm/xe_drm.h>
#include "xe_device.h" #include "xe_device.h"
...@@ -84,6 +85,21 @@ static int check_hw_engines(struct xe_device *xe, ...@@ -84,6 +85,21 @@ static int check_hw_engines(struct xe_device *xe,
DRM_XE_UFENCE_WAIT_VM_ERROR) DRM_XE_UFENCE_WAIT_VM_ERROR)
#define MAX_OP DRM_XE_UFENCE_WAIT_LTE #define MAX_OP DRM_XE_UFENCE_WAIT_LTE
static unsigned long to_jiffies_timeout(struct drm_xe_wait_user_fence *args)
{
unsigned long timeout;
if (args->flags & DRM_XE_UFENCE_WAIT_ABSTIME)
return drm_timeout_abs_to_jiffies(args->timeout);
if (args->timeout == MAX_SCHEDULE_TIMEOUT || args->timeout == 0)
return args->timeout;
timeout = nsecs_to_jiffies(args->timeout);
return timeout ?: 1;
}
int xe_wait_user_fence_ioctl(struct drm_device *dev, void *data, int xe_wait_user_fence_ioctl(struct drm_device *dev, void *data,
struct drm_file *file) struct drm_file *file)
{ {
...@@ -98,7 +114,8 @@ int xe_wait_user_fence_ioctl(struct drm_device *dev, void *data, ...@@ -98,7 +114,8 @@ int xe_wait_user_fence_ioctl(struct drm_device *dev, void *data,
int err; int err;
bool no_engines = args->flags & DRM_XE_UFENCE_WAIT_SOFT_OP || bool no_engines = args->flags & DRM_XE_UFENCE_WAIT_SOFT_OP ||
args->flags & DRM_XE_UFENCE_WAIT_VM_ERROR; args->flags & DRM_XE_UFENCE_WAIT_VM_ERROR;
unsigned long timeout = args->timeout; unsigned long timeout;
ktime_t start;
if (XE_IOCTL_ERR(xe, args->extensions) || XE_IOCTL_ERR(xe, args->pad) || if (XE_IOCTL_ERR(xe, args->extensions) || XE_IOCTL_ERR(xe, args->pad) ||
XE_IOCTL_ERR(xe, args->reserved[0] || args->reserved[1])) XE_IOCTL_ERR(xe, args->reserved[0] || args->reserved[1]))
...@@ -152,8 +169,18 @@ int xe_wait_user_fence_ioctl(struct drm_device *dev, void *data, ...@@ -152,8 +169,18 @@ int xe_wait_user_fence_ioctl(struct drm_device *dev, void *data,
addr = vm->async_ops.error_capture.addr; addr = vm->async_ops.error_capture.addr;
} }
if (XE_IOCTL_ERR(xe, timeout > MAX_SCHEDULE_TIMEOUT)) /*
return -EINVAL; * For negative timeout we want to wait "forever" by setting
* MAX_SCHEDULE_TIMEOUT. But we have to assign this value also
* to args->timeout to avoid being zeroed on the signal delivery
* (see arithmetics after wait).
*/
if (args->timeout < 0)
args->timeout = MAX_SCHEDULE_TIMEOUT;
timeout = to_jiffies_timeout(args);
start = ktime_get();
/* /*
* FIXME: Very simple implementation at the moment, single wait queue * FIXME: Very simple implementation at the moment, single wait queue
...@@ -192,17 +219,17 @@ int xe_wait_user_fence_ioctl(struct drm_device *dev, void *data, ...@@ -192,17 +219,17 @@ int xe_wait_user_fence_ioctl(struct drm_device *dev, void *data,
} else { } else {
remove_wait_queue(&xe->ufence_wq, &w_wait); remove_wait_queue(&xe->ufence_wq, &w_wait);
} }
if (!(args->flags & DRM_XE_UFENCE_WAIT_ABSTIME)) {
args->timeout -= ktime_to_ns(ktime_sub(ktime_get(), start));
if (args->timeout < 0)
args->timeout = 0;
}
if (XE_IOCTL_ERR(xe, err < 0)) if (XE_IOCTL_ERR(xe, err < 0))
return err; return err;
else if (XE_IOCTL_ERR(xe, !timeout)) else if (XE_IOCTL_ERR(xe, !timeout))
return -ETIME; return -ETIME;
/*
* Again very simple, return the time in jiffies that has past, may need
* a more precision
*/
if (args->flags & DRM_XE_UFENCE_WAIT_ABSTIME)
args->timeout = args->timeout - timeout;
return 0; return 0;
} }
...@@ -904,8 +904,20 @@ struct drm_xe_wait_user_fence { ...@@ -904,8 +904,20 @@ struct drm_xe_wait_user_fence {
#define DRM_XE_UFENCE_WAIT_U64 0xffffffffffffffffu #define DRM_XE_UFENCE_WAIT_U64 0xffffffffffffffffu
/** @mask: comparison mask */ /** @mask: comparison mask */
__u64 mask; __u64 mask;
/**
/** @timeout: how long to wait before bailing, value in jiffies */ * @timeout: how long to wait before bailing, value in nanoseconds.
* Without DRM_XE_UFENCE_WAIT_ABSTIME flag set (relative timeout)
* it contains timeout expressed in nanoseconds to wait (fence will
* expire at now() + timeout).
* When DRM_XE_UFENCE_WAIT_ABSTIME flat is set (absolute timeout) wait
* will end at timeout (uses system MONOTONIC_CLOCK).
* Passing negative timeout leads to neverending wait.
*
* On relative timeout this value is updated with timeout left
* (for restarting the call in case of signal delivery).
* On absolute timeout this value stays intact (restarted call still
* expire at the same point of time).
*/
__s64 timeout; __s64 timeout;
/** /**
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment