<div dir="ltr"><div class="gmail_quote"><div dir="ltr" class="gmail_attr">On Mon, Aug 22, 2022 at 8:27 AM Christian König <<a href="mailto:christian.koenig@amd.com">christian.koenig@amd.com</a>> wrote:<br></div><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex">Am 22.08.22 um 10:34 schrieb Bas Nieuwenhuizen:<br>
> On Mon, Aug 22, 2022 at 9:28 AM Dave Airlie <<a href="mailto:airlied@gmail.com" target="_blank">airlied@gmail.com</a>> wrote:<br>
>> On Mon, 22 Aug 2022 at 17:05, Dave Airlie <<a href="mailto:airlied@gmail.com" target="_blank">airlied@gmail.com</a>> wrote:<br>
>>> Hey,<br>
>>><br>
>>> I've just been looking at the vm bind type interfaces and wanted to at<br>
>>> least document how we think the unmapping API should work. I know I've<br>
>>> talked on irc before about this, but wanted to solidify things a bit<br>
>>> more around what is required vs what is a nice to have.<br>
>>><br>
>>> My main concerns/thoughts are around the unbind interfaces and how<br>
>>> close to munmap they should be.<br>
>>><br>
>>> I think the mapping operation is mostly consistent<br>
>>> MAP(bo handle, offset into bo, range, VM offset, VM flags)<br>
>>> which puts the range inside to bo at the offset in the current VM<br>
>>> (maybe take an optional vm_id).<br>
>>><br>
>>> now the simplest unmap I can see if one that parallel munmap<br>
>>> UNMAP(vmaddr, range);<br>
>>><br>
>>> But it begs the question on then how much the kernel needs to deal<br>
>>> with here, if we support random vmaddr,range then we really need to be<br>
>>> able to do everything munmap does for CPU VMA, which means splitting<br>
>>> ranges, joining ranges etc.<br>
>>><br>
>>> like<br>
>>> MAP(1, 0, 0x8000, 0xc0000)<br>
>>> UNMAP(0xc1000, 0x1000)<br>
>>> should that be possible?<br>
>>><br>
>>> Do we have any API usage (across Vulkan/CL/CUDA/ROCm etc) that<br>
>>> requires this sort of control, or should we be fine with only<br>
>>> unmapping objects exactly like how they were mapped in the first<br>
>>> place, and not have any splitting/joining?<br>
> Vulkan allows for this, though I haven't checked to what extent apps use it.<br>
<br>
This is massively used for partial resident textures under OpenGL as far <br>
as I know.<br>
<br>
E.g. you map a range like 1->10 as PRT and then then map real textures <br>
at 2, 5 and 7 or something like that.<br>
<br>
Saying that a functionality to map/enable PRT for a range is necessary <br>
as well. On amdgpu we have a special flag for that and in this case the <br>
BO to map can be NULL.<br></blockquote><div><br></div><div>NVIDIA has similar bits.  I don't remember if intel does or not.  Handling this as a map with BO=0 and a PRT flag of some sort seems like a perfectly reasonable way to handle it.<br></div><div> </div><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex">
> We could technically split all mapping/unmapping to be per single tile<br>
> in the userspace driver, which avoids the need for splitting/merging,<br>
> but that could very much be a pessimization.<br>
<br>
That would be pretty much a NAK from my side. A couple of hardware <br>
optimizations require mappings to be as large as possible.<br>
<br>
Otherwise we wouldn't be able to use huge/giant (2MiB, 1GiB) pages, <br>
power of two TLB reach optimizations (8KiB, 16KiB, 32KiB.....) as well <br>
as texture fetcher optimizations.<br></blockquote><div><br></div><div>Agreed.  Intel has such optimizations as well and they really do matter.  IDK about nvidia but I'd be surprised if they don't at least have a 2M variant or something.  Reducing page-table depth matters a lot for latency.<br></div><div> </div><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex">
>> I suppose it also asks the question around paralleling<br>
>><br>
>> fd = open()<br>
>> ptr = mmap(fd,)<br>
>> close(fd)<br>
>> the mapping is still valid.<br>
>><br>
>> I suppose our equiv is<br>
>> handle = bo_alloc()<br>
>> gpu_addr = vm_bind(handle,)<br>
>> gem_close(handle)<br>
>> is the gpu_addr still valid does the VM hold a reference on the kernel<br>
>> bo internally.<br>
> For Vulkan it looks like this is undefined and the above is not necessary:<br>
><br>
> "It is important to note that freeing a VkDeviceMemory object with<br>
> vkFreeMemory will not cause resources (or resource regions) bound to<br>
> the memory object to become unbound. Applications must not access<br>
> resources bound to memory that has been freed."<br>
> (32.7.6)<br></blockquote><div><br></div><div>I'm not sure about this particular question.  We need to be sure that maps get cleaned up eventually.  On the one hand, I think it's probably a valid API implementation to have each mapped page hold a reference similar to mmap and have vkDestroyImage or vkDestroyBuffer do an unmap to clean up the range.  However, clients may be surprised when they destroy a large memory object and can't reap the memory because of extra BO references they don't know about.  If BOs unmap themselves on close or if we had a way to take a VM+BO and say "unmap this BO from everywhere, please", we can clean up the memory pretty easily.  Without that, it's a giant PITA to do entirely inside the userspace driver because it requires us to globally track every mapping and that means data structures and locks.  Yes, such an ioctl would require the kernel to track things but the kernel is already tracking everything that's bound, so hopefully it doesn't add much.</div><div><br></div><div>--Jason<br></div><div> </div><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex">
Additional to what was discussed here so far we need an array on in and <br>
out drm_syncobj for both map as well as unmap.<br>
<br>
E.g. when the mapping/unmapping should happen and when it is completed <br>
etc...<br>
<br>
Christian.<br>
<br>
><br>
><br>
>> Dave.<br>
>>> Dave.<br>
<br>
</blockquote></div></div>