+cc some more Intel people On Wed, Mar 04, 2020 at 01:36:03AM +0200, Jarkko Sakkinen wrote: > +static struct page *sgx_encl_get_backing_page(struct sgx_encl *encl, > + pgoff_t index) > +{ > + struct inode *inode = encl->backing->f_path.dentry->d_inode; > + struct address_space *mapping = inode->i_mapping; > + gfp_t gfpmask = mapping_gfp_mask(mapping); Question for folks that care about cgroup accounting: should the shmem page allocation be accounted, even though it'll charge the "wrong" task? E.g. gfp_t gfpmask = mapping_gfp_mask(mapping) | __GFP_ACCOUNT; Background details... To reclaim an EPC page, the encrypted data must be written to addressable memory when the page is evicted from the EPC. I.e. RAM or persistent memory acts as swap space for the EPC. In the SGX code this is referred to as the "backing store" or just "backing". As implemented, the backing for EPC reclaim is not pre-allocated. I don't think anyone would want to change this behavior as it would eat regular memory even if EPC isn't oversubscribed. But, as implemented it's not accounted. The reason we haven't explicitly forced accounting is because allocations for the backing would be charged to the task doing the evicting, as opposed to the task that originally allocated the EPC page. This means that task 'A' can do a DOS of sorts on task 'B' by allocating a large amount of EPC with the goal of causing 'B' to trigger (normal memory) OOM and get killed. AFAICT, that's still preferable to not accounting at all, e.g. at least 'B' would need to be allocating a decent amount of EPC as well to trigger OOM, as opposed to today where the allocations are completely unaccounted and so cause a more global OOM. We've also discussed taking a file descriptor to hold the backing, but unless I'm misreading the pagecache code, that doesn't solve the incorrect accounting problem because the current task, i.e. evicting task, would be charged. In other words, whether the backing is kernel or user controlled is purely an ABI question. In theory, SGX could do some form of manual memcg accounting in sgx_encl_get_backing_page(), but that gets ridiculously ugly, if it's even possible, e.g. determining whether or not the page was already allocated would be a nightmare. Long term, I think this can be solved, or at least mitigated, by adding a "backing" limit to the EPC cgroup (which obviously doesn't exist in yet) to allow limiting the amount of memory that can be indirectly consumed by creating a large enclave. E.g. account each page during ENCLAVE_ADD_PAGES, with some logic to transfer the charges when a mm_struct is disassociated from an enclave (which will probably be needed for the base EPC cgroup anyways). > + > + return shmem_read_mapping_page_gfp(mapping, index, gfpmask); > +} > + > +/** > + * sgx_encl_get_backing() - Pin the backing storage > + * @encl: an enclave > + * @page_index: enclave page index > + * @backing: data for accessing backing storage for the page > + * > + * Pin the backing storage pages for storing the encrypted contents and Paging > + * Crypto MetaData (PCMD) of an enclave page. > + * > + * Return: > + * 0 on success, > + * -errno otherwise. > + */ > +int sgx_encl_get_backing(struct sgx_encl *encl, unsigned long page_index, > + struct sgx_backing *backing) > +{ > + pgoff_t pcmd_index = PFN_DOWN(encl->size) + 1 + (page_index >> 5); > + struct page *contents; > + struct page *pcmd; > + > + contents = sgx_encl_get_backing_page(encl, page_index); > + if (IS_ERR(contents)) > + return PTR_ERR(contents); > + > + pcmd = sgx_encl_get_backing_page(encl, pcmd_index); > + if (IS_ERR(pcmd)) { > + put_page(contents); > + return PTR_ERR(pcmd); > + } > + > + backing->page_index = page_index; > + backing->contents = contents; > + backing->pcmd = pcmd; > + backing->pcmd_offset = > + (page_index & (PAGE_SIZE / sizeof(struct sgx_pcmd) - 1)) * > + sizeof(struct sgx_pcmd); > + > + return 0; > +} > + > +/** > + * sgx_encl_put_backing() - Unpin the backing storage > + * @backing: data for accessing backing storage for the page > + * @do_write: mark pages dirty > + */ > +void sgx_encl_put_backing(struct sgx_backing *backing, bool do_write) > +{ > + if (do_write) { > + set_page_dirty(backing->pcmd); > + set_page_dirty(backing->contents); > + } > + > + put_page(backing->pcmd); > + put_page(backing->contents); > +}