Message ID | 20181015230515.3695-1-keithp@keithp.com (mailing list archive) |
---|---|
State | New, archived |
Headers | show |
Series | vulkan: Add VK_EXT_calibrated_timestamps extension (radv and anv) [v3] | expand |
On Mon, Oct 15, 2018 at 6:05 PM Keith Packard <keithp@keithp.com> wrote: > Offers three clocks, device, clock monotonic and clock monotonic > raw. Could use some kernel support to reduce the deviation between > clock values. > > v2: > Ensure deviation is at least as big as the GPU time interval. > > v3: > Set device->lost when returning DEVICE_LOST. > Use MAX2 and DIV_ROUND_UP instead of open coding these. > Delete spurious TIMESTAMP in radv version. > Suggested-by: Jason Ekstrand <jason@jlekstrand.net> > Suggested-by: Lionel Landwerlin <lionel.g.landwerlin@intel.com> > > Signed-off-by: Keith Packard <keithp@keithp.com> > --- > src/amd/vulkan/radv_device.c | 81 +++++++++++++++++++++++++++ > src/amd/vulkan/radv_extensions.py | 1 + > src/intel/vulkan/anv_device.c | 89 ++++++++++++++++++++++++++++++ > src/intel/vulkan/anv_extensions.py | 1 + > src/intel/vulkan/anv_gem.c | 13 +++++ > src/intel/vulkan/anv_private.h | 2 + > 6 files changed, 187 insertions(+) > > diff --git a/src/amd/vulkan/radv_device.c b/src/amd/vulkan/radv_device.c > index 174922780fc..80050485e54 100644 > --- a/src/amd/vulkan/radv_device.c > +++ b/src/amd/vulkan/radv_device.c > @@ -4955,3 +4955,84 @@ radv_GetDeviceGroupPeerMemoryFeatures( > VK_PEER_MEMORY_FEATURE_GENERIC_SRC_BIT | > VK_PEER_MEMORY_FEATURE_GENERIC_DST_BIT; > } > + > +static const VkTimeDomainEXT radv_time_domains[] = { > + VK_TIME_DOMAIN_DEVICE_EXT, > + VK_TIME_DOMAIN_CLOCK_MONOTONIC_EXT, > + VK_TIME_DOMAIN_CLOCK_MONOTONIC_RAW_EXT, > +}; > + > +VkResult radv_GetPhysicalDeviceCalibrateableTimeDomainsEXT( > + VkPhysicalDevice physicalDevice, > + uint32_t *pTimeDomainCount, > + VkTimeDomainEXT *pTimeDomains) > +{ > + int d; > + VK_OUTARRAY_MAKE(out, pTimeDomains, pTimeDomainCount); > + > + for (d = 0; d < ARRAY_SIZE(radv_time_domains); d++) { > + vk_outarray_append(&out, i) { > + *i = radv_time_domains[d]; > + } > + } > + > + return vk_outarray_status(&out); > +} > + > +static uint64_t > +radv_clock_gettime(clockid_t clock_id) > +{ > + struct timespec current; > + int ret; > + > + ret = clock_gettime(clock_id, ¤t); > + if (ret < 0 && clock_id == CLOCK_MONOTONIC_RAW) > + ret = clock_gettime(CLOCK_MONOTONIC, ¤t); > + if (ret < 0) > + return 0; > + > + return (uint64_t) current.tv_sec * 1000000000ULL + current.tv_nsec; > +} > + > +VkResult radv_GetCalibratedTimestampsEXT( > + VkDevice _device, > + uint32_t timestampCount, > + const VkCalibratedTimestampInfoEXT *pTimestampInfos, > + uint64_t *pTimestamps, > + uint64_t *pMaxDeviation) > +{ > + RADV_FROM_HANDLE(radv_device, device, _device); > + uint32_t clock_crystal_freq = > device->physical_device->rad_info.clock_crystal_freq; > + int d; > + uint64_t begin, end; > + > + begin = radv_clock_gettime(CLOCK_MONOTONIC_RAW); > + > + for (d = 0; d < timestampCount; d++) { > + switch (pTimestampInfos[d].timeDomain) { > + case VK_TIME_DOMAIN_DEVICE_EXT: > + pTimestamps[d] = > device->ws->query_value(device->ws, > + > RADEON_TIMESTAMP); > + break; > + case VK_TIME_DOMAIN_CLOCK_MONOTONIC_EXT: > + pTimestamps[d] = > radv_clock_gettime(CLOCK_MONOTONIC); > + break; > + > + case VK_TIME_DOMAIN_CLOCK_MONOTONIC_RAW_EXT: > + pTimestamps[d] = begin; > + break; > + default: > + pTimestamps[d] = 0; > + break; > + } > + } > + > + end = radv_clock_gettime(CLOCK_MONOTONIC_RAW); > + > + uint64_t clock_period = end - begin; > + uint64_t device_period = DIV_ROUND_UP(1000000, clock_crystal_freq); > + > + *pMaxDeviation = MAX2(clock_period, device_period); > + > + return VK_SUCCESS; > +} > diff --git a/src/amd/vulkan/radv_extensions.py > b/src/amd/vulkan/radv_extensions.py > index 5dcedae1c63..4c81d3f0068 100644 > --- a/src/amd/vulkan/radv_extensions.py > +++ b/src/amd/vulkan/radv_extensions.py > @@ -92,6 +92,7 @@ EXTENSIONS = [ > Extension('VK_KHR_display', 23, > 'VK_USE_PLATFORM_DISPLAY_KHR'), > Extension('VK_EXT_direct_mode_display', 1, > 'VK_USE_PLATFORM_DISPLAY_KHR'), > Extension('VK_EXT_acquire_xlib_display', 1, > 'VK_USE_PLATFORM_XLIB_XRANDR_EXT'), > + Extension('VK_EXT_calibrated_timestamps', 1, True), > Extension('VK_EXT_conditional_rendering', 1, True), > Extension('VK_EXT_conservative_rasterization', 1, > 'device->rad_info.chip_class >= GFX9'), > Extension('VK_EXT_display_surface_counter', 1, > 'VK_USE_PLATFORM_DISPLAY_KHR'), > diff --git a/src/intel/vulkan/anv_device.c b/src/intel/vulkan/anv_device.c > index a2551452eb1..249d7f36f0f 100644 > --- a/src/intel/vulkan/anv_device.c > +++ b/src/intel/vulkan/anv_device.c > @@ -3021,6 +3021,95 @@ void anv_DestroyFramebuffer( > vk_free2(&device->alloc, pAllocator, fb); > } > > +static const VkTimeDomainEXT anv_time_domains[] = { > + VK_TIME_DOMAIN_DEVICE_EXT, > + VK_TIME_DOMAIN_CLOCK_MONOTONIC_EXT, > + VK_TIME_DOMAIN_CLOCK_MONOTONIC_RAW_EXT, > +}; > + > +VkResult anv_GetPhysicalDeviceCalibrateableTimeDomainsEXT( > + VkPhysicalDevice physicalDevice, > + uint32_t *pTimeDomainCount, > + VkTimeDomainEXT *pTimeDomains) > +{ > + int d; > + VK_OUTARRAY_MAKE(out, pTimeDomains, pTimeDomainCount); > + > + for (d = 0; d < ARRAY_SIZE(anv_time_domains); d++) { > + vk_outarray_append(&out, i) { > + *i = anv_time_domains[d]; > + } > + } > + > + return vk_outarray_status(&out); > +} > + > +static uint64_t > +anv_clock_gettime(clockid_t clock_id) > +{ > + struct timespec current; > + int ret; > + > + ret = clock_gettime(clock_id, ¤t); > + if (ret < 0 && clock_id == CLOCK_MONOTONIC_RAW) > + ret = clock_gettime(CLOCK_MONOTONIC, ¤t); > + if (ret < 0) > + return 0; > + > + return (uint64_t) current.tv_sec * 1000000000ULL + current.tv_nsec; > +} > + > +#define TIMESTAMP 0x2358 > + > +VkResult anv_GetCalibratedTimestampsEXT( > + VkDevice _device, > + uint32_t timestampCount, > + const VkCalibratedTimestampInfoEXT *pTimestampInfos, > + uint64_t *pTimestamps, > + uint64_t *pMaxDeviation) > +{ > + ANV_FROM_HANDLE(anv_device, device, _device); > + uint64_t timestamp_frequency = device->info.timestamp_frequency; > + int ret; > + int d; > + uint64_t begin, end; > + > + begin = anv_clock_gettime(CLOCK_MONOTONIC_RAW); > + > + for (d = 0; d < timestampCount; d++) { > + switch (pTimestampInfos[d].timeDomain) { > + case VK_TIME_DOMAIN_DEVICE_EXT: > + ret = anv_gem_reg_read(device, TIMESTAMP | 1, > + &pTimestamps[d]); > + > + if (ret != 0) { > + device->lost = TRUE; > + return VK_ERROR_DEVICE_LOST; > + } > + break; > + case VK_TIME_DOMAIN_CLOCK_MONOTONIC_EXT: > + pTimestamps[d] = anv_clock_gettime(CLOCK_MONOTONIC); > + break; > + > + case VK_TIME_DOMAIN_CLOCK_MONOTONIC_RAW_EXT: > + pTimestamps[d] = begin; > + break; > + default: > + pTimestamps[d] = 0; > + break; > + } > + } > + > + end = anv_clock_gettime(CLOCK_MONOTONIC_RAW); > + > + uint64_t clock_period = end - begin; > + uint64_t device_period = DIV_ROUND_UP(1000000000, timestamp_frequency); > + > + *pMaxDeviation = MAX2(clock_period, device_period); > + > + return VK_SUCCESS; > +} > + > /* vk_icd.h does not declare this function, so we declare it here to > * suppress Wmissing-prototypes. > */ > diff --git a/src/intel/vulkan/anv_extensions.py > b/src/intel/vulkan/anv_extensions.py > index d4915c95013..a8535964da7 100644 > --- a/src/intel/vulkan/anv_extensions.py > +++ b/src/intel/vulkan/anv_extensions.py > @@ -126,6 +126,7 @@ EXTENSIONS = [ > Extension('VK_EXT_vertex_attribute_divisor', 3, True), > Extension('VK_EXT_post_depth_coverage', 1, > 'device->info.gen >= 9'), > Extension('VK_EXT_sampler_filter_minmax', 1, > 'device->info.gen >= 9'), > + Extension('VK_EXT_calibrated_timestamps', 1, True), > ] > > class VkVersion: > diff --git a/src/intel/vulkan/anv_gem.c b/src/intel/vulkan/anv_gem.c > index c43b5ef9e06..1bdf040c1a3 100644 > --- a/src/intel/vulkan/anv_gem.c > +++ b/src/intel/vulkan/anv_gem.c > @@ -423,6 +423,19 @@ anv_gem_fd_to_handle(struct anv_device *device, int > fd) > return args.handle; > } > > +int > +anv_gem_reg_read(struct anv_device *device, uint32_t offset, uint64_t > *result) > +{ > + struct drm_i915_reg_read args = { > + .offset = offset > + }; > + > + int ret = anv_ioctl(device->fd, DRM_IOCTL_I915_REG_READ, &args); > + > + *result = args.val; > + return ret; > +} > You need to add this to anv_gem_stubs.c as well or else the unit tests won't build. Sorry for not catching it earlier. I'm always missing this too. With that fixed, the anv bits are Reviewed-by: Jason Ekstrand <jason@jlekstrand.net> > + > #ifndef SYNC_IOC_MAGIC > /* duplicated from linux/sync_file.h to avoid build-time dependency > * on new (v4.7) kernel headers. Once distro's are mostly using > diff --git a/src/intel/vulkan/anv_private.h > b/src/intel/vulkan/anv_private.h > index 599b903f25c..08376b00c8e 100644 > --- a/src/intel/vulkan/anv_private.h > +++ b/src/intel/vulkan/anv_private.h > @@ -1103,6 +1103,8 @@ int anv_gem_get_aperture(int fd, uint64_t *size); > int anv_gem_gpu_get_reset_stats(struct anv_device *device, > uint32_t *active, uint32_t *pending); > int anv_gem_handle_to_fd(struct anv_device *device, uint32_t gem_handle); > +int anv_gem_reg_read(struct anv_device *device, > + uint32_t offset, uint64_t *result); > uint32_t anv_gem_fd_to_handle(struct anv_device *device, int fd); > int anv_gem_set_caching(struct anv_device *device, uint32_t gem_handle, > uint32_t caching); > int anv_gem_set_domain(struct anv_device *device, uint32_t gem_handle, > -- > 2.19.1 > > _______________________________________________ > mesa-dev mailing list > mesa-dev@lists.freedesktop.org > https://lists.freedesktop.org/mailman/listinfo/mesa-dev > <div dir="ltr"><div class="gmail_quote"><div dir="ltr">On Mon, Oct 15, 2018 at 6:05 PM Keith Packard <<a href="mailto:keithp@keithp.com" target="_blank">keithp@keithp.com</a>> wrote:<br></div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">Offers three clocks, device, clock monotonic and clock monotonic<br> raw. Could use some kernel support to reduce the deviation between<br> clock values.<br> <br> v2:<br> Ensure deviation is at least as big as the GPU time interval.<br> <br> v3:<br> Set device->lost when returning DEVICE_LOST.<br> Use MAX2 and DIV_ROUND_UP instead of open coding these.<br> Delete spurious TIMESTAMP in radv version.<br> Suggested-by: Jason Ekstrand <<a href="mailto:jason@jlekstrand.net" target="_blank">jason@jlekstrand.net</a>><br> Suggested-by: Lionel Landwerlin <<a href="mailto:lionel.g.landwerlin@intel.com" target="_blank">lionel.g.landwerlin@intel.com</a>><br> <br> Signed-off-by: Keith Packard <<a href="mailto:keithp@keithp.com" target="_blank">keithp@keithp.com</a>><br> ---<br> src/amd/vulkan/radv_device.c | 81 +++++++++++++++++++++++++++<br> src/amd/vulkan/radv_extensions.py | 1 +<br> src/intel/vulkan/anv_device.c | 89 ++++++++++++++++++++++++++++++<br> src/intel/vulkan/anv_extensions.py | 1 +<br> src/intel/vulkan/anv_gem.c | 13 +++++<br> src/intel/vulkan/anv_private.h | 2 +<br> 6 files changed, 187 insertions(+)<br> <br> diff --git a/src/amd/vulkan/radv_device.c b/src/amd/vulkan/radv_device.c<br> index 174922780fc..80050485e54 100644<br> --- a/src/amd/vulkan/radv_device.c<br> +++ b/src/amd/vulkan/radv_device.c<br> @@ -4955,3 +4955,84 @@ radv_GetDeviceGroupPeerMemoryFeatures(<br> VK_PEER_MEMORY_FEATURE_GENERIC_SRC_BIT |<br> VK_PEER_MEMORY_FEATURE_GENERIC_DST_BIT;<br> }<br> +<br> +static const VkTimeDomainEXT radv_time_domains[] = {<br> + VK_TIME_DOMAIN_DEVICE_EXT,<br> + VK_TIME_DOMAIN_CLOCK_MONOTONIC_EXT,<br> + VK_TIME_DOMAIN_CLOCK_MONOTONIC_RAW_EXT,<br> +};<br> +<br> +VkResult radv_GetPhysicalDeviceCalibrateableTimeDomainsEXT(<br> + VkPhysicalDevice physicalDevice,<br> + uint32_t *pTimeDomainCount,<br> + VkTimeDomainEXT *pTimeDomains)<br> +{<br> + int d;<br> + VK_OUTARRAY_MAKE(out, pTimeDomains, pTimeDomainCount);<br> +<br> + for (d = 0; d < ARRAY_SIZE(radv_time_domains); d++) {<br> + vk_outarray_append(&out, i) {<br> + *i = radv_time_domains[d];<br> + }<br> + }<br> +<br> + return vk_outarray_status(&out);<br> +}<br> +<br> +static uint64_t<br> +radv_clock_gettime(clockid_t clock_id)<br> +{<br> + struct timespec current;<br> + int ret;<br> +<br> + ret = clock_gettime(clock_id, &current);<br> + if (ret < 0 && clock_id == CLOCK_MONOTONIC_RAW)<br> + ret = clock_gettime(CLOCK_MONOTONIC, &current);<br> + if (ret < 0)<br> + return 0;<br> +<br> + return (uint64_t) current.tv_sec * 1000000000ULL + current.tv_nsec;<br> +}<br> +<br> +VkResult radv_GetCalibratedTimestampsEXT(<br> + VkDevice _device,<br> + uint32_t timestampCount,<br> + const VkCalibratedTimestampInfoEXT *pTimestampInfos,<br> + uint64_t *pTimestamps,<br> + uint64_t *pMaxDeviation)<br> +{<br> + RADV_FROM_HANDLE(radv_device, device, _device);<br> + uint32_t clock_crystal_freq = device->physical_device->rad_info.clock_crystal_freq;<br> + int d;<br> + uint64_t begin, end;<br> +<br> + begin = radv_clock_gettime(CLOCK_MONOTONIC_RAW);<br> +<br> + for (d = 0; d < timestampCount; d++) {<br> + switch (pTimestampInfos[d].timeDomain) {<br> + case VK_TIME_DOMAIN_DEVICE_EXT:<br> + pTimestamps[d] = device->ws->query_value(device->ws,<br> + RADEON_TIMESTAMP);<br> + break;<br> + case VK_TIME_DOMAIN_CLOCK_MONOTONIC_EXT:<br> + pTimestamps[d] = radv_clock_gettime(CLOCK_MONOTONIC);<br> + break;<br> +<br> + case VK_TIME_DOMAIN_CLOCK_MONOTONIC_RAW_EXT:<br> + pTimestamps[d] = begin;<br> + break;<br> + default:<br> + pTimestamps[d] = 0;<br> + break;<br> + }<br> + }<br> +<br> + end = radv_clock_gettime(CLOCK_MONOTONIC_RAW);<br> +<br> + uint64_t clock_period = end - begin;<br> + uint64_t device_period = DIV_ROUND_UP(1000000, clock_crystal_freq);<br> +<br> + *pMaxDeviation = MAX2(clock_period, device_period);<br> +<br> + return VK_SUCCESS;<br> +}<br> diff --git a/src/amd/vulkan/radv_extensions.py b/src/amd/vulkan/radv_extensions.py<br> index 5dcedae1c63..4c81d3f0068 100644<br> --- a/src/amd/vulkan/radv_extensions.py<br> +++ b/src/amd/vulkan/radv_extensions.py<br> @@ -92,6 +92,7 @@ EXTENSIONS = [<br> Extension('VK_KHR_display', 23, 'VK_USE_PLATFORM_DISPLAY_KHR'),<br> Extension('VK_EXT_direct_mode_display', 1, 'VK_USE_PLATFORM_DISPLAY_KHR'),<br> Extension('VK_EXT_acquire_xlib_display', 1, 'VK_USE_PLATFORM_XLIB_XRANDR_EXT'),<br> + Extension('VK_EXT_calibrated_timestamps', 1, True),<br> Extension('VK_EXT_conditional_rendering', 1, True),<br> Extension('VK_EXT_conservative_rasterization', 1, 'device->rad_info.chip_class >= GFX9'),<br> Extension('VK_EXT_display_surface_counter', 1, 'VK_USE_PLATFORM_DISPLAY_KHR'),<br> diff --git a/src/intel/vulkan/anv_device.c b/src/intel/vulkan/anv_device.c<br> index a2551452eb1..249d7f36f0f 100644<br> --- a/src/intel/vulkan/anv_device.c<br> +++ b/src/intel/vulkan/anv_device.c<br> @@ -3021,6 +3021,95 @@ void anv_DestroyFramebuffer(<br> vk_free2(&device->alloc, pAllocator, fb);<br> }<br> <br> +static const VkTimeDomainEXT anv_time_domains[] = {<br> + VK_TIME_DOMAIN_DEVICE_EXT,<br> + VK_TIME_DOMAIN_CLOCK_MONOTONIC_EXT,<br> + VK_TIME_DOMAIN_CLOCK_MONOTONIC_RAW_EXT,<br> +};<br> +<br> +VkResult anv_GetPhysicalDeviceCalibrateableTimeDomainsEXT(<br> + VkPhysicalDevice physicalDevice,<br> + uint32_t *pTimeDomainCount,<br> + VkTimeDomainEXT *pTimeDomains)<br> +{<br> + int d;<br> + VK_OUTARRAY_MAKE(out, pTimeDomains, pTimeDomainCount);<br> +<br> + for (d = 0; d < ARRAY_SIZE(anv_time_domains); d++) {<br> + vk_outarray_append(&out, i) {<br> + *i = anv_time_domains[d];<br> + }<br> + }<br> +<br> + return vk_outarray_status(&out);<br> +}<br> +<br> +static uint64_t<br> +anv_clock_gettime(clockid_t clock_id)<br> +{<br> + struct timespec current;<br> + int ret;<br> +<br> + ret = clock_gettime(clock_id, &current);<br> + if (ret < 0 && clock_id == CLOCK_MONOTONIC_RAW)<br> + ret = clock_gettime(CLOCK_MONOTONIC, &current);<br> + if (ret < 0)<br> + return 0;<br> +<br> + return (uint64_t) current.tv_sec * 1000000000ULL + current.tv_nsec;<br> +}<br> +<br> +#define TIMESTAMP 0x2358<br> +<br> +VkResult anv_GetCalibratedTimestampsEXT(<br> + VkDevice _device,<br> + uint32_t timestampCount,<br> + const VkCalibratedTimestampInfoEXT *pTimestampInfos,<br> + uint64_t *pTimestamps,<br> + uint64_t *pMaxDeviation)<br> +{<br> + ANV_FROM_HANDLE(anv_device, device, _device);<br> + uint64_t timestamp_frequency = device->info.timestamp_frequency;<br> + int ret;<br> + int d;<br> + uint64_t begin, end;<br> +<br> + begin = anv_clock_gettime(CLOCK_MONOTONIC_RAW);<br> +<br> + for (d = 0; d < timestampCount; d++) {<br> + switch (pTimestampInfos[d].timeDomain) {<br> + case VK_TIME_DOMAIN_DEVICE_EXT:<br> + ret = anv_gem_reg_read(device, TIMESTAMP | 1,<br> + &pTimestamps[d]);<br> +<br> + if (ret != 0) {<br> + device->lost = TRUE;<br> + return VK_ERROR_DEVICE_LOST;<br> + }<br> + break;<br> + case VK_TIME_DOMAIN_CLOCK_MONOTONIC_EXT:<br> + pTimestamps[d] = anv_clock_gettime(CLOCK_MONOTONIC);<br> + break;<br> +<br> + case VK_TIME_DOMAIN_CLOCK_MONOTONIC_RAW_EXT:<br> + pTimestamps[d] = begin;<br> + break;<br> + default:<br> + pTimestamps[d] = 0;<br> + break;<br> + }<br> + }<br> +<br> + end = anv_clock_gettime(CLOCK_MONOTONIC_RAW);<br> +<br> + uint64_t clock_period = end - begin;<br> + uint64_t device_period = DIV_ROUND_UP(1000000000, timestamp_frequency);<br> +<br> + *pMaxDeviation = MAX2(clock_period, device_period);<br> +<br> + return VK_SUCCESS;<br> +}<br> +<br> /* vk_icd.h does not declare this function, so we declare it here to<br> * suppress Wmissing-prototypes.<br> */<br> diff --git a/src/intel/vulkan/anv_extensions.py b/src/intel/vulkan/anv_extensions.py<br> index d4915c95013..a8535964da7 100644<br> --- a/src/intel/vulkan/anv_extensions.py<br> +++ b/src/intel/vulkan/anv_extensions.py<br> @@ -126,6 +126,7 @@ EXTENSIONS = [<br> Extension('VK_EXT_vertex_attribute_divisor', 3, True),<br> Extension('VK_EXT_post_depth_coverage', 1, 'device->info.gen >= 9'),<br> Extension('VK_EXT_sampler_filter_minmax', 1, 'device->info.gen >= 9'),<br> + Extension('VK_EXT_calibrated_timestamps', 1, True),<br> ]<br> <br> class VkVersion:<br> diff --git a/src/intel/vulkan/anv_gem.c b/src/intel/vulkan/anv_gem.c<br> index c43b5ef9e06..1bdf040c1a3 100644<br> --- a/src/intel/vulkan/anv_gem.c<br> +++ b/src/intel/vulkan/anv_gem.c<br> @@ -423,6 +423,19 @@ anv_gem_fd_to_handle(struct anv_device *device, int fd)<br> return args.handle;<br> }<br> <br> +int<br> +anv_gem_reg_read(struct anv_device *device, uint32_t offset, uint64_t *result)<br> +{<br> + struct drm_i915_reg_read args = {<br> + .offset = offset<br> + };<br> +<br> + int ret = anv_ioctl(device->fd, DRM_IOCTL_I915_REG_READ, &args);<br> +<br> + *result = args.val;<br> + return ret;<br> +}<br></blockquote></div><div class="gmail_quote"><br></div><div class="gmail_quote">You need to add this to anv_gem_stubs.c as well or else the unit tests won't build. Sorry for not catching it earlier. I'm always missing this too.</div><div class="gmail_quote"><br></div><div class="gmail_quote">With that fixed, the anv bits are</div><div class="gmail_quote"><br></div><div class="gmail_quote">Reviewed-by: Jason Ekstrand <<a href="mailto:jason@jlekstrand.net">jason@jlekstrand.net</a>><br></div><div class="gmail_quote"><div> </div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex"> +<br> #ifndef SYNC_IOC_MAGIC<br> /* duplicated from linux/sync_file.h to avoid build-time dependency<br> * on new (v4.7) kernel headers. Once distro's are mostly using<br> diff --git a/src/intel/vulkan/anv_private.h b/src/intel/vulkan/anv_private.h<br> index 599b903f25c..08376b00c8e 100644<br> --- a/src/intel/vulkan/anv_private.h<br> +++ b/src/intel/vulkan/anv_private.h<br> @@ -1103,6 +1103,8 @@ int anv_gem_get_aperture(int fd, uint64_t *size);<br> int anv_gem_gpu_get_reset_stats(struct anv_device *device,<br> uint32_t *active, uint32_t *pending);<br> int anv_gem_handle_to_fd(struct anv_device *device, uint32_t gem_handle);<br> +int anv_gem_reg_read(struct anv_device *device,<br> + uint32_t offset, uint64_t *result);<br> uint32_t anv_gem_fd_to_handle(struct anv_device *device, int fd);<br> int anv_gem_set_caching(struct anv_device *device, uint32_t gem_handle, uint32_t caching);<br> int anv_gem_set_domain(struct anv_device *device, uint32_t gem_handle,<br> -- <br> 2.19.1<br> <br> _______________________________________________<br> mesa-dev mailing list<br> <a href="mailto:mesa-dev@lists.freedesktop.org" target="_blank">mesa-dev@lists.freedesktop.org</a><br> <a href="https://lists.freedesktop.org/mailman/listinfo/mesa-dev" rel="noreferrer" target="_blank">https://lists.freedesktop.org/mailman/listinfo/mesa-dev</a><br> </blockquote></div></div>
Jason Ekstrand <jason@jlekstrand.net> writes: > You need to add this to anv_gem_stubs.c as well or else the unit tests > won't build. Sorry for not catching it earlier. I'm always missing this > too. Well, that's a bit hard to test as -Dbuild-tests=true fails in a bunch of glx tests, but I think I've got it. > With that fixed, the anv bits are > > Reviewed-by: Jason Ekstrand <jason@jlekstrand.net> Thanks. I haven't heard from any radv developers, so I can either split the patch apart or wait for another day or two. In any case, I'll post v4 of the patch here with the anv_gem_reg_read addition made.
diff --git a/src/amd/vulkan/radv_device.c b/src/amd/vulkan/radv_device.c index 174922780fc..80050485e54 100644 --- a/src/amd/vulkan/radv_device.c +++ b/src/amd/vulkan/radv_device.c @@ -4955,3 +4955,84 @@ radv_GetDeviceGroupPeerMemoryFeatures( VK_PEER_MEMORY_FEATURE_GENERIC_SRC_BIT | VK_PEER_MEMORY_FEATURE_GENERIC_DST_BIT; } + +static const VkTimeDomainEXT radv_time_domains[] = { + VK_TIME_DOMAIN_DEVICE_EXT, + VK_TIME_DOMAIN_CLOCK_MONOTONIC_EXT, + VK_TIME_DOMAIN_CLOCK_MONOTONIC_RAW_EXT, +}; + +VkResult radv_GetPhysicalDeviceCalibrateableTimeDomainsEXT( + VkPhysicalDevice physicalDevice, + uint32_t *pTimeDomainCount, + VkTimeDomainEXT *pTimeDomains) +{ + int d; + VK_OUTARRAY_MAKE(out, pTimeDomains, pTimeDomainCount); + + for (d = 0; d < ARRAY_SIZE(radv_time_domains); d++) { + vk_outarray_append(&out, i) { + *i = radv_time_domains[d]; + } + } + + return vk_outarray_status(&out); +} + +static uint64_t +radv_clock_gettime(clockid_t clock_id) +{ + struct timespec current; + int ret; + + ret = clock_gettime(clock_id, ¤t); + if (ret < 0 && clock_id == CLOCK_MONOTONIC_RAW) + ret = clock_gettime(CLOCK_MONOTONIC, ¤t); + if (ret < 0) + return 0; + + return (uint64_t) current.tv_sec * 1000000000ULL + current.tv_nsec; +} + +VkResult radv_GetCalibratedTimestampsEXT( + VkDevice _device, + uint32_t timestampCount, + const VkCalibratedTimestampInfoEXT *pTimestampInfos, + uint64_t *pTimestamps, + uint64_t *pMaxDeviation) +{ + RADV_FROM_HANDLE(radv_device, device, _device); + uint32_t clock_crystal_freq = device->physical_device->rad_info.clock_crystal_freq; + int d; + uint64_t begin, end; + + begin = radv_clock_gettime(CLOCK_MONOTONIC_RAW); + + for (d = 0; d < timestampCount; d++) { + switch (pTimestampInfos[d].timeDomain) { + case VK_TIME_DOMAIN_DEVICE_EXT: + pTimestamps[d] = device->ws->query_value(device->ws, + RADEON_TIMESTAMP); + break; + case VK_TIME_DOMAIN_CLOCK_MONOTONIC_EXT: + pTimestamps[d] = radv_clock_gettime(CLOCK_MONOTONIC); + break; + + case VK_TIME_DOMAIN_CLOCK_MONOTONIC_RAW_EXT: + pTimestamps[d] = begin; + break; + default: + pTimestamps[d] = 0; + break; + } + } + + end = radv_clock_gettime(CLOCK_MONOTONIC_RAW); + + uint64_t clock_period = end - begin; + uint64_t device_period = DIV_ROUND_UP(1000000, clock_crystal_freq); + + *pMaxDeviation = MAX2(clock_period, device_period); + + return VK_SUCCESS; +} diff --git a/src/amd/vulkan/radv_extensions.py b/src/amd/vulkan/radv_extensions.py index 5dcedae1c63..4c81d3f0068 100644 --- a/src/amd/vulkan/radv_extensions.py +++ b/src/amd/vulkan/radv_extensions.py @@ -92,6 +92,7 @@ EXTENSIONS = [ Extension('VK_KHR_display', 23, 'VK_USE_PLATFORM_DISPLAY_KHR'), Extension('VK_EXT_direct_mode_display', 1, 'VK_USE_PLATFORM_DISPLAY_KHR'), Extension('VK_EXT_acquire_xlib_display', 1, 'VK_USE_PLATFORM_XLIB_XRANDR_EXT'), + Extension('VK_EXT_calibrated_timestamps', 1, True), Extension('VK_EXT_conditional_rendering', 1, True), Extension('VK_EXT_conservative_rasterization', 1, 'device->rad_info.chip_class >= GFX9'), Extension('VK_EXT_display_surface_counter', 1, 'VK_USE_PLATFORM_DISPLAY_KHR'), diff --git a/src/intel/vulkan/anv_device.c b/src/intel/vulkan/anv_device.c index a2551452eb1..249d7f36f0f 100644 --- a/src/intel/vulkan/anv_device.c +++ b/src/intel/vulkan/anv_device.c @@ -3021,6 +3021,95 @@ void anv_DestroyFramebuffer( vk_free2(&device->alloc, pAllocator, fb); } +static const VkTimeDomainEXT anv_time_domains[] = { + VK_TIME_DOMAIN_DEVICE_EXT, + VK_TIME_DOMAIN_CLOCK_MONOTONIC_EXT, + VK_TIME_DOMAIN_CLOCK_MONOTONIC_RAW_EXT, +}; + +VkResult anv_GetPhysicalDeviceCalibrateableTimeDomainsEXT( + VkPhysicalDevice physicalDevice, + uint32_t *pTimeDomainCount, + VkTimeDomainEXT *pTimeDomains) +{ + int d; + VK_OUTARRAY_MAKE(out, pTimeDomains, pTimeDomainCount); + + for (d = 0; d < ARRAY_SIZE(anv_time_domains); d++) { + vk_outarray_append(&out, i) { + *i = anv_time_domains[d]; + } + } + + return vk_outarray_status(&out); +} + +static uint64_t +anv_clock_gettime(clockid_t clock_id) +{ + struct timespec current; + int ret; + + ret = clock_gettime(clock_id, ¤t); + if (ret < 0 && clock_id == CLOCK_MONOTONIC_RAW) + ret = clock_gettime(CLOCK_MONOTONIC, ¤t); + if (ret < 0) + return 0; + + return (uint64_t) current.tv_sec * 1000000000ULL + current.tv_nsec; +} + +#define TIMESTAMP 0x2358 + +VkResult anv_GetCalibratedTimestampsEXT( + VkDevice _device, + uint32_t timestampCount, + const VkCalibratedTimestampInfoEXT *pTimestampInfos, + uint64_t *pTimestamps, + uint64_t *pMaxDeviation) +{ + ANV_FROM_HANDLE(anv_device, device, _device); + uint64_t timestamp_frequency = device->info.timestamp_frequency; + int ret; + int d; + uint64_t begin, end; + + begin = anv_clock_gettime(CLOCK_MONOTONIC_RAW); + + for (d = 0; d < timestampCount; d++) { + switch (pTimestampInfos[d].timeDomain) { + case VK_TIME_DOMAIN_DEVICE_EXT: + ret = anv_gem_reg_read(device, TIMESTAMP | 1, + &pTimestamps[d]); + + if (ret != 0) { + device->lost = TRUE; + return VK_ERROR_DEVICE_LOST; + } + break; + case VK_TIME_DOMAIN_CLOCK_MONOTONIC_EXT: + pTimestamps[d] = anv_clock_gettime(CLOCK_MONOTONIC); + break; + + case VK_TIME_DOMAIN_CLOCK_MONOTONIC_RAW_EXT: + pTimestamps[d] = begin; + break; + default: + pTimestamps[d] = 0; + break; + } + } + + end = anv_clock_gettime(CLOCK_MONOTONIC_RAW); + + uint64_t clock_period = end - begin; + uint64_t device_period = DIV_ROUND_UP(1000000000, timestamp_frequency); + + *pMaxDeviation = MAX2(clock_period, device_period); + + return VK_SUCCESS; +} + /* vk_icd.h does not declare this function, so we declare it here to * suppress Wmissing-prototypes. */ diff --git a/src/intel/vulkan/anv_extensions.py b/src/intel/vulkan/anv_extensions.py index d4915c95013..a8535964da7 100644 --- a/src/intel/vulkan/anv_extensions.py +++ b/src/intel/vulkan/anv_extensions.py @@ -126,6 +126,7 @@ EXTENSIONS = [ Extension('VK_EXT_vertex_attribute_divisor', 3, True), Extension('VK_EXT_post_depth_coverage', 1, 'device->info.gen >= 9'), Extension('VK_EXT_sampler_filter_minmax', 1, 'device->info.gen >= 9'), + Extension('VK_EXT_calibrated_timestamps', 1, True), ] class VkVersion: diff --git a/src/intel/vulkan/anv_gem.c b/src/intel/vulkan/anv_gem.c index c43b5ef9e06..1bdf040c1a3 100644 --- a/src/intel/vulkan/anv_gem.c +++ b/src/intel/vulkan/anv_gem.c @@ -423,6 +423,19 @@ anv_gem_fd_to_handle(struct anv_device *device, int fd) return args.handle; } +int +anv_gem_reg_read(struct anv_device *device, uint32_t offset, uint64_t *result) +{ + struct drm_i915_reg_read args = { + .offset = offset + }; + + int ret = anv_ioctl(device->fd, DRM_IOCTL_I915_REG_READ, &args); + + *result = args.val; + return ret; +} + #ifndef SYNC_IOC_MAGIC /* duplicated from linux/sync_file.h to avoid build-time dependency * on new (v4.7) kernel headers. Once distro's are mostly using diff --git a/src/intel/vulkan/anv_private.h b/src/intel/vulkan/anv_private.h index 599b903f25c..08376b00c8e 100644 --- a/src/intel/vulkan/anv_private.h +++ b/src/intel/vulkan/anv_private.h @@ -1103,6 +1103,8 @@ int anv_gem_get_aperture(int fd, uint64_t *size); int anv_gem_gpu_get_reset_stats(struct anv_device *device, uint32_t *active, uint32_t *pending); int anv_gem_handle_to_fd(struct anv_device *device, uint32_t gem_handle); +int anv_gem_reg_read(struct anv_device *device, + uint32_t offset, uint64_t *result); uint32_t anv_gem_fd_to_handle(struct anv_device *device, int fd); int anv_gem_set_caching(struct anv_device *device, uint32_t gem_handle, uint32_t caching); int anv_gem_set_domain(struct anv_device *device, uint32_t gem_handle,
Offers three clocks, device, clock monotonic and clock monotonic raw. Could use some kernel support to reduce the deviation between clock values. v2: Ensure deviation is at least as big as the GPU time interval. v3: Set device->lost when returning DEVICE_LOST. Use MAX2 and DIV_ROUND_UP instead of open coding these. Delete spurious TIMESTAMP in radv version. Suggested-by: Jason Ekstrand <jason@jlekstrand.net> Suggested-by: Lionel Landwerlin <lionel.g.landwerlin@intel.com> Signed-off-by: Keith Packard <keithp@keithp.com> --- src/amd/vulkan/radv_device.c | 81 +++++++++++++++++++++++++++ src/amd/vulkan/radv_extensions.py | 1 + src/intel/vulkan/anv_device.c | 89 ++++++++++++++++++++++++++++++ src/intel/vulkan/anv_extensions.py | 1 + src/intel/vulkan/anv_gem.c | 13 +++++ src/intel/vulkan/anv_private.h | 2 + 6 files changed, 187 insertions(+)