On Tue, Dec 04, 2018 at 09:06:59AM -0800, Andi Kleen wrote: > jglisse@xxxxxxxxxx writes: > > > + > > +To help with forward compatibility each object as a version value and > > +it is mandatory for user space to only use target or initiator with > > +version supported by the user space. For instance if user space only > > +knows about what version 1 means and sees a target with version 2 then > > +the user space must ignore that target as if it does not exist. > > So once v2 is introduced all applications that only support v1 break. > > That seems very un-Linux and will break Linus' "do not break existing > applications" rule. > > The standard approach that if you add something incompatible is to > add new field, but keep the old ones. No that's not how it is suppose to work. So let says it is 2018 and you have v1 memory (like your regular main DDR memory for instance) then it will always be expose a v1 memory. Fast forward 2020 and you have this new type of memory that is not cache coherent and you want to expose this to userspace through HMS. What you do is a kernel patch that introduce the v2 type for target and define a set of new sysfs file to describe what v2 is. On this new computer you report your usual main memory as v1 and your new memory as v2. So the application that only knew about v1 will keep using any v1 memory on your new platform but it will not use any of the new memory v2 which is what you want to happen. You do not have to break existing application while allowing to add new type of memory. Sorry if it was unclear. I will try to reformulate and give an example as above. > > +2) hbind() bind range of virtual address to heterogeneous memory > > +================================================================ > > + > > +So instead of using a bitmap, hbind() take an array of uid and each uid > > +is a unique memory target inside the new memory topology description. > > You didn't define what an uid is? > > user id ? > > Please use sensible terminology that doesn't conflict with existing > usages. > > I assume it's some kind of number that identifies a node in your > graph. Correct uid is unique id given to each node in the graph. I will clarify that. > > +User space also provide an array of modifiers. Modifier can be seen as > > +the flags parameter of mbind() but here we use an array so that user > > +space can not only supply a modifier but also value with it. This should > > +allow the API to grow more features in the future. Kernel should return > > +-EINVAL if it is provided with an unkown modifier and just ignore the > > +call all together, forcing the user space to restrict itself to modifier > > +supported by the kernel it is running on (i know i am dreaming about well > > +behave user space). > > It sounds like you're trying to define a system call with built in > ioctl? Is that really a good idea? > > If you need ioctl you know where to find it. Well i would like to get thing running in the wild with some guinea pig user to get feedback from end user. It would be easier if i can do this with upstream kernel and not some random branch in my private repo. While doing that i would like to avoid commiting to a syscall upstream. So the way i see around this is doing a driver under staging with an ioctl which will be turn into a syscall once some confidence into the API is gain. If you think i should do a syscall right away i am not against doing that. > > Please don't over design APIs like this. So there is 2 approach here. I can define 2 syscall, one for migration and one for policy. Migration and policy are 2 different thing from all existing user point of view. By defining 2 syscall i can cut them down to do this one thing and one thing only and make it as simple and lean as possible. In the present version i took the other approach of defining just one API that can grow to do more thing. I know the unix way is one simple tool for one simple job. I can switch to the simple call for one action. > > +3) Tracking and applying heterogeneous memory policies > > +====================================================== > > + > > +Current memory policy infrastructure is node oriented, instead of > > +changing that and risking breakage and regression HMS adds a new > > +heterogeneous policy tracking infra-structure. The expectation is > > +that existing application can keep using mbind() and all existing > > +infrastructure under-disturb and unaffected, while new application > > +will use the new API and should avoid mix and matching both (as they > > +can achieve the same thing with the new API). > > I think we need a stronger motivation to define a completely > parallel and somewhat redundant infrastructure. What breakage > are you worried about? Some memory expose through HMS is not allocated by regular memory allocator. For instance GPU memory is manage by GPU driver, so when you want to use GPU memory (either as a policy or by migrating to it) you need to use the GPU allocator to allocate that memory. HMS adds a bunch of callback to target structure so that device driver can expose a generic API to core kernel to do such allocation. Now i can change existing code path to use target structure as an intermediary for allocation but this is changing hot code path and i doubt it would be welcome today. Eventually i think we will want that to happen and can work on minimizing cost for user that do not use thing like GPU. The transition phase will take times (couple years) and i would like to avoid disturbing existing workload while we migrate GPU user to this new API. > The obvious alternative would of course be to add some extra > enumeration to the existing nodes. We can not extend NUMA node to expose GPU memory. GPU memory on current AMD and Intel platform is not cache coherent and thus should not be use for random memory allocation. It should really stay a thing user have to explicitly select to use. Note that the useage we have here is that when you use GPU memory it is as if the range of virtual address is swapped out from CPU point of view but the GPU can access it. > It's a strange document. It goes from very high level to low level > with nothing inbetween. I think you need a lot more details > in the middle, in particularly how these new interfaces > should be used. For example how should an application > know how to look for a specific type of device? > How is an automated tool supposed to use the enumeration? > etc. Today user use dedicated API (OpenCL, ROCm, CUDA, ...) those high level API all have the API i present here in one form or another. So i want to move this high level API that is actively use by program today into the kernel. The end game is to create common infrastructure for various accelerator hardware (GPU, FPGA, ...) to manage memory. This is something ask by end user for one simple reasons. Today users have to mix and match multiple API in their application and when they want to exchange data between one device that use one API and another device that use another API they have to do explicit copy and rebuild their data structure inside the new memory. When you move over thing like tree or any complex data structure you have to rebuilt it ie redo the pointers link between the nodes of your data structure. This is highly error prone complex and wasteful (you have to burn CPU cycles to do that). Now if you can use the same address space as all the other memory allocation in your program and move data around from one device to another with a common API that works on all the various devices, you are eliminating that complex step and making the end user life much easier. So i am doing this to help existing users by addressing an issues that is becoming harder and harder to solve for userspace. My end game is to blur the boundary between CPU and device like GPU, FPGA, ... Thank you for taking time to read this proposal and for your feed- back. Much appreciated. I will try to include your comments in my v2. Cheers, Jérôme