|
45 | 45 | #include <linux/workqueue.h>
|
46 | 46 | #include <linux/ratelimit.h>
|
47 | 47 | #include <linux/moduleparam.h>
|
| 48 | +#ifdef CONFIG_XEN_GRANT_DMA_ALLOC |
| 49 | +#include <linux/dma-mapping.h> |
| 50 | +#endif |
48 | 51 |
|
49 | 52 | #include <xen/xen.h>
|
50 | 53 | #include <xen/interface/xen.h>
|
|
57 | 60 | #ifdef CONFIG_X86
|
58 | 61 | #include <asm/xen/cpuid.h>
|
59 | 62 | #endif
|
| 63 | +#include <xen/mem-reservation.h> |
60 | 64 | #include <asm/xen/hypercall.h>
|
61 | 65 | #include <asm/xen/interface.h>
|
62 | 66 |
|
@@ -838,6 +842,99 @@ void gnttab_free_pages(int nr_pages, struct page **pages)
|
838 | 842 | }
|
839 | 843 | EXPORT_SYMBOL_GPL(gnttab_free_pages);
|
840 | 844 |
|
| 845 | +#ifdef CONFIG_XEN_GRANT_DMA_ALLOC |
| 846 | +/** |
| 847 | + * gnttab_dma_alloc_pages - alloc DMAable pages suitable for grant mapping into |
| 848 | + * @args: arguments to the function |
| 849 | + */ |
| 850 | +int gnttab_dma_alloc_pages(struct gnttab_dma_alloc_args *args) |
| 851 | +{ |
| 852 | + unsigned long pfn, start_pfn; |
| 853 | + size_t size; |
| 854 | + int i, ret; |
| 855 | + |
| 856 | + size = args->nr_pages << PAGE_SHIFT; |
| 857 | + if (args->coherent) |
| 858 | + args->vaddr = dma_alloc_coherent(args->dev, size, |
| 859 | + &args->dev_bus_addr, |
| 860 | + GFP_KERNEL | __GFP_NOWARN); |
| 861 | + else |
| 862 | + args->vaddr = dma_alloc_wc(args->dev, size, |
| 863 | + &args->dev_bus_addr, |
| 864 | + GFP_KERNEL | __GFP_NOWARN); |
| 865 | + if (!args->vaddr) { |
| 866 | + pr_debug("Failed to allocate DMA buffer of size %zu\n", size); |
| 867 | + return -ENOMEM; |
| 868 | + } |
| 869 | + |
| 870 | + start_pfn = __phys_to_pfn(args->dev_bus_addr); |
| 871 | + for (pfn = start_pfn, i = 0; pfn < start_pfn + args->nr_pages; |
| 872 | + pfn++, i++) { |
| 873 | + struct page *page = pfn_to_page(pfn); |
| 874 | + |
| 875 | + args->pages[i] = page; |
| 876 | + args->frames[i] = xen_page_to_gfn(page); |
| 877 | + xenmem_reservation_scrub_page(page); |
| 878 | + } |
| 879 | + |
| 880 | + xenmem_reservation_va_mapping_reset(args->nr_pages, args->pages); |
| 881 | + |
| 882 | + ret = xenmem_reservation_decrease(args->nr_pages, args->frames); |
| 883 | + if (ret != args->nr_pages) { |
| 884 | + pr_debug("Failed to decrease reservation for DMA buffer\n"); |
| 885 | + ret = -EFAULT; |
| 886 | + goto fail; |
| 887 | + } |
| 888 | + |
| 889 | + ret = gnttab_pages_set_private(args->nr_pages, args->pages); |
| 890 | + if (ret < 0) |
| 891 | + goto fail; |
| 892 | + |
| 893 | + return 0; |
| 894 | + |
| 895 | +fail: |
| 896 | + gnttab_dma_free_pages(args); |
| 897 | + return ret; |
| 898 | +} |
| 899 | +EXPORT_SYMBOL_GPL(gnttab_dma_alloc_pages); |
| 900 | + |
| 901 | +/** |
| 902 | + * gnttab_dma_free_pages - free DMAable pages |
| 903 | + * @args: arguments to the function |
| 904 | + */ |
| 905 | +int gnttab_dma_free_pages(struct gnttab_dma_alloc_args *args) |
| 906 | +{ |
| 907 | + size_t size; |
| 908 | + int i, ret; |
| 909 | + |
| 910 | + gnttab_pages_clear_private(args->nr_pages, args->pages); |
| 911 | + |
| 912 | + for (i = 0; i < args->nr_pages; i++) |
| 913 | + args->frames[i] = page_to_xen_pfn(args->pages[i]); |
| 914 | + |
| 915 | + ret = xenmem_reservation_increase(args->nr_pages, args->frames); |
| 916 | + if (ret != args->nr_pages) { |
| 917 | + pr_debug("Failed to decrease reservation for DMA buffer\n"); |
| 918 | + ret = -EFAULT; |
| 919 | + } else { |
| 920 | + ret = 0; |
| 921 | + } |
| 922 | + |
| 923 | + xenmem_reservation_va_mapping_update(args->nr_pages, args->pages, |
| 924 | + args->frames); |
| 925 | + |
| 926 | + size = args->nr_pages << PAGE_SHIFT; |
| 927 | + if (args->coherent) |
| 928 | + dma_free_coherent(args->dev, size, |
| 929 | + args->vaddr, args->dev_bus_addr); |
| 930 | + else |
| 931 | + dma_free_wc(args->dev, size, |
| 932 | + args->vaddr, args->dev_bus_addr); |
| 933 | + return ret; |
| 934 | +} |
| 935 | +EXPORT_SYMBOL_GPL(gnttab_dma_free_pages); |
| 936 | +#endif |
| 937 | + |
841 | 938 | /* Handling of paged out grant targets (GNTST_eagain) */
|
842 | 939 | #define MAX_DELAY 256
|
843 | 940 | static inline void
|
|
0 commit comments