[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index] Re: [Xen-devel] [PATCH 3/4] xen/arm: introduce XENMEM_cache_flush
>>> On 02.10.14 at 12:02, <stefano.stabellini@xxxxxxxxxxxxx> wrote: > --- a/xen/arch/arm/mm.c > +++ b/xen/arch/arm/mm.c > @@ -1134,6 +1134,98 @@ long arch_memory_op(int op, > XEN_GUEST_HANDLE_PARAM(void) arg) > case XENMEM_get_sharing_shared_pages: > case XENMEM_get_sharing_freed_pages: > return 0; > + case XENMEM_cache_flush: Blank line missing above this one. > + { > + struct xen_cache_flush cflush; > + struct domain *d, *owner; > + struct page_info *page; > + uint64_t mfn, end; > + uint64_t offset, size; > + void *v; > + int ret = 0; > + > + if ( copy_from_guest(&cflush, arg, 1) ) > + return -EFAULT; > + > + d = rcu_lock_current_domain(); > + if ( d == NULL ) > + return -ESRCH; I think this should be the last check prior to the loop below, simplifying error handling (break instead of goto). > + > + if ( (cflush.size >> PAGE_SHIFT) > (1U<<MAX_ORDER) ) The why is size a uint64_t in the first place? > + { > + printk(XENLOG_G_ERR "invalid size %llx\n", cflush.size); > + ret = -EINVAL; > + goto out; > + } > + > + if ( cflush.size == 0 || cflush.op == 0 ) > + { > + ret = 0; > + goto out; > + } > + > + if ( cflush.op & ~(XENMEM_CACHE_INVAL|XENMEM_CACHE_CLEAN) ) > + { > + printk(XENLOG_G_ERR "invalid op %x\n", cflush.op); > + ret = -EINVAL; > + goto out; > + } > + > + end = cflush.addr + cflush.size; As said in various contexts before: Such an addition can overflow, and (even if this doesn't matter here due to the earlier check) you can't express "the entire address space" with a (base,size) pair either. Representing arbitrary ranges is better done using inclusive pairs of addresses/MFNs/whatever. > + while ( cflush.addr < end ) > + { > + mfn = cflush.addr >> PAGE_SHIFT; > + offset = cflush.addr & ~PAGE_MASK; > + > + if ( !mfn_valid(mfn) ) > + { > + printk(XENLOG_G_ERR "mfn=%llx is not valid\n", mfn); > + ret = -EINVAL; > + goto out; > + } > + > + page = mfn_to_page(mfn); > + if ( !page ) > + { > + printk(XENLOG_G_ERR "couldn't get page for mfn %llx\n", mfn); > + ret = -EFAULT; > + goto out; > + } > + > + owner = page_get_owner(page); > + if ( !owner ) > + { > + printk(XENLOG_G_ERR "couldn't get owner for mfn %llx\n", > mfn); > + ret = -EFAULT; > + goto out; > + } > + > + if ( owner != d && !grant_map_exists(d, owner->grant_table, mfn) > ) > + { > + printk(XENLOG_G_ERR "mfn %llx hasn't been granted by %d to > %d\n", mfn, owner->domain_id, d->domain_id); Long line. But - do you really need all these printk()s? > + ret = -EINVAL; > + goto out; > + } > + > + v = map_domain_page(mfn); > + v += offset; > + size = cflush.size - cflush.addr; > + if ( size > PAGE_SIZE - offset ) > + size = PAGE_SIZE - offset; > + > + if ( cflush.op & XENMEM_CACHE_INVAL ) > + invalidate_xen_dcache_va_range(v, size); > + if ( cflush.op & XENMEM_CACHE_CLEAN ) > + clean_xen_dcache_va_range(v, size); Is this really the right sequence when both flags are set? Of course I can only guess a what "clean" and "invalidate" here mean (nothing more specific is being said alongside their definition), but I'd expect you to want to clean (flush) cache contents _before_ invalidating. > + unmap_domain_page(v); > + > + cflush.addr += PAGE_SIZE - offset; > + } > + > +out: Labels should be indented by at least one space. But you don't really need the label anyway if you follow the suggestion above. Jan _______________________________________________ Xen-devel mailing list Xen-devel@xxxxxxxxxxxxx http://lists.xen.org/xen-devel
|
Lists.xenproject.org is hosted with RackSpace, monitoring our |