mfn = _mfn(gpfn);
}
+ else if ( is_domain_using_staticmem(d) )
+ {
+ /*
+ * No easy way to guarantee the retrieved pages are contiguous,
+ * so forbid non-zero-order requests here.
+ */
+ if ( a->extent_order != 0 )
+ {
+ gdprintk(XENLOG_WARNING,
+ "Cannot allocate static order-%u pages for %pd\n",
+ a->extent_order, d);
+ goto out;
+ }
+
+ mfn = acquire_reserved_page(d, a->memflags);
+ if ( mfn_eq(mfn, INVALID_MFN) )
+ {
+ gdprintk(XENLOG_WARNING,
+ "%pd: failed to retrieve a reserved page\n",
+ d);
+ goto out;
+ }
+ }
else
{
page = alloc_domheap_pages(d, a->extent_order, a->memflags);
put_domain(d);
}
-static bool __init prepare_staticmem_pages(struct page_info *pg,
- unsigned long nr_mfns,
- unsigned int memflags)
+static bool prepare_staticmem_pages(struct page_info *pg, unsigned long nr_mfns,
+ unsigned int memflags)
{
bool need_tlbflush = false;
uint32_t tlbflush_timestamp = 0;
return pg;
}
+static int assign_domstatic_pages(struct domain *d, struct page_info *pg,
+ unsigned int nr_mfns, unsigned int memflags)
+{
+ if ( !d || (memflags & (MEMF_no_owner | MEMF_no_refcount)) )
+ {
+ /*
+ * Respective handling omitted here because right now
+ * acquired static memory is only for domain's RAM.
+ */
+ ASSERT_UNREACHABLE();
+ return -EINVAL;
+ }
+
+ if ( assign_pages(pg, nr_mfns, d, memflags) )
+ return -EINVAL;
+
+ return 0;
+}
+
/*
* Acquire nr_mfns contiguous pages, starting at #smfn, of static memory,
* then assign them to one specific domain #d.
if ( !pg )
return -ENOENT;
- if ( !d || (memflags & (MEMF_no_owner | MEMF_no_refcount)) )
- {
- /*
- * Respective handling omitted here because right now
- * acquired static memory is only for guest RAM.
- */
- ASSERT_UNREACHABLE();
- return -EINVAL;
- }
-
- if ( assign_pages(pg, nr_mfns, d, memflags) )
+ if ( assign_domstatic_pages(d, pg, nr_mfns, memflags) )
{
unprepare_staticmem_pages(pg, nr_mfns, memflags & MEMF_no_scrub);
return -EINVAL;
return 0;
}
+
+/*
+ * Acquire a page from reserved page list(resv_page_list), when populating
+ * memory for static domain on runtime.
+ */
+mfn_t acquire_reserved_page(struct domain *d, unsigned int memflags)
+{
+ struct page_info *page;
+
+ ASSERT_ALLOC_CONTEXT();
+
+ /* Acquire a page from reserved page list(resv_page_list). */
+ spin_lock(&d->page_alloc_lock);
+ page = page_list_remove_head(&d->resv_page_list);
+ spin_unlock(&d->page_alloc_lock);
+ if ( unlikely(!page) )
+ return INVALID_MFN;
+
+ if ( !prepare_staticmem_pages(page, 1, memflags) )
+ goto fail;
+
+ if ( assign_domstatic_pages(d, page, 1, memflags) )
+ goto fail_assign;
+
+ return page_to_mfn(page);
+
+ fail_assign:
+ /*
+ * The page was never accessible by the domain. So scrubbing can be
+ * skipped
+ */
+ unprepare_staticmem_pages(page, 1, false);
+ fail:
+ spin_lock(&d->page_alloc_lock);
+ page_list_add_tail(page, &d->resv_page_list);
+ spin_unlock(&d->page_alloc_lock);
+ return INVALID_MFN;
+}
#endif
/*