kernel/0102-refactor-MMZ-vb-and-heap-MMZ-code-moved-in-kernel.patch

5937 lines
162 KiB
Diff
Raw Normal View History

2024-12-15 18:29:23 +00:00
From 4b762de47916d2f2cd54071cad33b163b0df8448 Mon Sep 17 00:00:00 2001
From: linmin <linmin@eswincomputing.com>
Date: Wed, 19 Jun 2024 16:23:06 +0800
Subject: [PATCH 102/219] refactor(MMZ vb and heap):MMZ code moved in kernel
Changelogs:
1.Moved es_buddy source code to drivers/memory/eswin/es_buddy
2.Moved es_proc source code to drivers/memory/eswin/es_proc
3.Moved mmz_vb source code to drivers/memory/eswin/es_mmz_vb
4.Moved es_rsvmem_heap source code to drivers/memory/eswin/es_rsvmem_heap
Signed-off-by: linmin <linmin@eswincomputing.com>
Reviewed-by: ningyu <ningyu@eswincomputing.com>
---
arch/riscv/configs/win2030_defconfig | 4 +
drivers/memory/eswin/Kconfig | 4 +
drivers/memory/eswin/Makefile | 8 +
drivers/memory/eswin/buddy.h | 44 +-
drivers/memory/eswin/es_buddy/Kconfig | 6 +
drivers/memory/eswin/es_buddy/Makefile | 3 +
.../eswin/es_buddy/buddy_allocator/buddy.c | 222 ++
drivers/memory/eswin/es_buddy/es_buddy.c | 186 ++
drivers/memory/eswin/es_buddy/es_buddy.h | 52 +
drivers/memory/eswin/es_mmz_vb/Kconfig | 6 +
drivers/memory/eswin/es_mmz_vb/Makefile | 5 +
.../eswin/es_mmz_vb/include/linux/mmz_vb.h | 58 +
.../es_mmz_vb/include/linux/mmz_vb_type.h | 17 +
drivers/memory/eswin/es_mmz_vb/mmz_vb.c | 2375 +++++++++++++++++
drivers/memory/eswin/es_proc/Kconfig | 7 +
drivers/memory/eswin/es_proc/Makefile | 7 +
drivers/memory/eswin/es_proc/es_proc.c | 233 ++
.../eswin/es_proc/include/linux/es_proc.h | 40 +
drivers/memory/eswin/es_rsvmem_heap/Kconfig | 6 +
drivers/memory/eswin/es_rsvmem_heap/Makefile | 8 +
.../dmabuf-heap-import-helper.c | 652 +++++
.../es_rsvmem_heap/eswin_rsvmem_common.c | 447 ++++
.../eswin/es_rsvmem_heap/eswin_rsvmem_heap.c | 634 +++++
.../include/linux/mem_perf_api.h | 123 +
.../include/uapi/linux/eswin_rsvmem_common.h | 56 +
include/linux/dmabuf-heap-import-helper.h | 100 +
include/linux/eswin_rsvmem_common.h | 94 +
include/uapi/linux/es_vb_user.h | 58 +
include/uapi/linux/mmz_vb.h | 175 ++
29 files changed, 5606 insertions(+), 24 deletions(-)
create mode 100644 drivers/memory/eswin/es_buddy/Kconfig
create mode 100644 drivers/memory/eswin/es_buddy/Makefile
create mode 100644 drivers/memory/eswin/es_buddy/buddy_allocator/buddy.c
create mode 100644 drivers/memory/eswin/es_buddy/es_buddy.c
create mode 100644 drivers/memory/eswin/es_buddy/es_buddy.h
create mode 100644 drivers/memory/eswin/es_mmz_vb/Kconfig
create mode 100644 drivers/memory/eswin/es_mmz_vb/Makefile
create mode 100644 drivers/memory/eswin/es_mmz_vb/include/linux/mmz_vb.h
create mode 100644 drivers/memory/eswin/es_mmz_vb/include/linux/mmz_vb_type.h
create mode 100644 drivers/memory/eswin/es_mmz_vb/mmz_vb.c
create mode 100644 drivers/memory/eswin/es_proc/Kconfig
create mode 100644 drivers/memory/eswin/es_proc/Makefile
create mode 100644 drivers/memory/eswin/es_proc/es_proc.c
create mode 100644 drivers/memory/eswin/es_proc/include/linux/es_proc.h
create mode 100644 drivers/memory/eswin/es_rsvmem_heap/Kconfig
create mode 100644 drivers/memory/eswin/es_rsvmem_heap/Makefile
create mode 100644 drivers/memory/eswin/es_rsvmem_heap/dmabuf-heap-import-helper.c
create mode 100644 drivers/memory/eswin/es_rsvmem_heap/eswin_rsvmem_common.c
create mode 100644 drivers/memory/eswin/es_rsvmem_heap/eswin_rsvmem_heap.c
create mode 100644 drivers/memory/eswin/es_rsvmem_heap/include/linux/mem_perf_api.h
create mode 100644 drivers/memory/eswin/es_rsvmem_heap/include/uapi/linux/eswin_rsvmem_common.h
create mode 100644 include/linux/dmabuf-heap-import-helper.h
create mode 100644 include/linux/eswin_rsvmem_common.h
create mode 100644 include/uapi/linux/es_vb_user.h
create mode 100644 include/uapi/linux/mmz_vb.h
diff --git a/arch/riscv/configs/win2030_defconfig b/arch/riscv/configs/win2030_defconfig
index 370b8dbe4a2e..32b31b28bdbd 100644
--- a/arch/riscv/configs/win2030_defconfig
+++ b/arch/riscv/configs/win2030_defconfig
@@ -736,6 +736,10 @@ CONFIG_RPMSG_VIRTIO=y
CONFIG_ARCH_ESWIN_EIC770X_SOC_FAMILY=y
CONFIG_EXTCON=y
CONFIG_MEMORY=y
+CONFIG_ESWIN_BUDDY=y
+CONFIG_ESWIN_PROC=y
+CONFIG_ESWIN_RSVMEM_HEAP=y
+CONFIG_ESWIN_MMZ_VB=y
CONFIG_PWM=y
CONFIG_PWM_ESWIN=y
CONFIG_RESET_ESWIN_WIN2030=y
diff --git a/drivers/memory/eswin/Kconfig b/drivers/memory/eswin/Kconfig
index 8f85e3cb466b..013e2f376dd3 100644
--- a/drivers/memory/eswin/Kconfig
+++ b/drivers/memory/eswin/Kconfig
@@ -25,5 +25,9 @@ config ESWIN_RSV_MEMBLOCK
If unsure, say "n".
source "drivers/memory/eswin/codacache/Kconfig"
+source "drivers/memory/eswin/es_buddy/Kconfig"
+source "drivers/memory/eswin/es_proc/Kconfig"
+source "drivers/memory/eswin/es_rsvmem_heap/Kconfig"
+source "drivers/memory/eswin/es_mmz_vb/Kconfig"
endif
diff --git a/drivers/memory/eswin/Makefile b/drivers/memory/eswin/Makefile
index 1b732b2a439d..3e5e09bab5dd 100644
--- a/drivers/memory/eswin/Makefile
+++ b/drivers/memory/eswin/Makefile
@@ -2,3 +2,11 @@
obj-$(CONFIG_ESWIN_MC) += eswin_cpuid_hartid_convert.o
obj-$(CONFIG_ESWIN_RSV_MEMBLOCK) += eswin_memblock.o
obj-$(CONFIG_ESWIN_CODACACHE_CONTROLLER) += codacache/
+obj-$(CONFIG_ESWIN_BUDDY) += es_buddy/
+obj-$(CONFIG_ESWIN_PROC) += es_proc/
+obj-$(CONFIG_ESWIN_RSVMEM_HEAP) += es_rsvmem_heap/
+obj-$(CONFIG_ESWIN_RSVMEM_HEAP) += es_mmz_vb/
+
+ES_MEM_HEADER := drivers/memory/eswin/
+
+COPY_HEADERS := $(shell cp $(ES_MEM_HEADER)/*.h include/linux)
\ No newline at end of file
diff --git a/drivers/memory/eswin/buddy.h b/drivers/memory/eswin/buddy.h
index 2c40d1116ad8..ff47325f2d27 100644
--- a/drivers/memory/eswin/buddy.h
+++ b/drivers/memory/eswin/buddy.h
@@ -1,3 +1,13 @@
+// SPDX-License-Identifier: GPL-2.0-only
+/*
+ * Header file of ESWIN internal buddy allocator
+ *
+ * Copyright 2024 Beijing ESWIN Computing Technology Co., Ltd.
+ * Authors:
+ * LinMin<linmin@eswincomputing.com>
+ *
+ */
+
#ifndef __BUDDY_H__
#define __BUDDY_H__
@@ -38,13 +48,11 @@
#define es_spin_lock(esLock)
#define es_spin_unlock(esLock)
#endif
-/*
- * <20><><EFBFBD>Page<67><65><EFBFBD><EFBFBD><EFBFBD><EFBFBD>״̬
- * */
+
enum esPageflags_e{
- enPG_head, //<2F><><EFBFBD><EFBFBD>buddyϵͳ<CFB5>ڣ<EFBFBD><DAA3>׸<EFBFBD>ҳ
- enPG_tail, //<2F><><EFBFBD><EFBFBD>buddyϵͳ<CFB5>ڣ<EFBFBD><DAA3><EFBFBD>ҳ֮<D2B3><D6AE><EFBFBD>ҳ<EFBFBD><D2B3>
- enPG_buddy, //<2F><>buddyϵͳ<CFB5><CDB3>
+ enPG_head,
+ enPG_tail,
+ enPG_buddy,
};
#define BUDDY_PAGE_SHIFT PAGE_SHIFT//(12UL)
@@ -110,11 +118,7 @@ void buddy_free_pages(struct mem_zone *zone,
struct esPage_s *page);
unsigned long buddy_num_free_page(struct mem_zone *zone);
-/*
- * ҳ<><D2B3>Ϊ<EFBFBD><CEAA><EFBFBD>һ<E0A3BA><D2BB><EFBFBD>ǵ<EFBFBD>ҳ<EFBFBD><D2B3>zero page<67><65>,
- * һ<><D2BB><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ҳ<EFBFBD><D2B3>compound page<67><65><EFBFBD><EFBFBD>
- * <20><><EFBFBD>ҳ<EFBFBD>ĵ<EFBFBD>һ<EFBFBD><D2BB><EFBFBD><EFBFBD>head<61><64><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ϊtail<69><6C>
- * */
+
static inline void __esSetPageHead(struct esPage_s *page)
{
page->flags |= (1UL<<enPG_head);
@@ -160,9 +164,7 @@ static inline int esPageBuddy(struct esPage_s *page)
return (page->flags & (1UL<<enPG_buddy));
}
-/*
- * <20><><EFBFBD><EFBFBD>ҳ<EFBFBD><D2B3>order<65><72>PG_buddy<64><79>־
- * */
+
static inline void set_page_order_buddy(struct esPage_s *page, unsigned long order)
{
page->order = order;
@@ -175,9 +177,7 @@ static inline void rmv_page_order_buddy(struct esPage_s *page)
__esClearPageBuddy(page);
}
-/*
- * <20><><EFBFBD><EFBFBD>buddyҳ
- * */
+
static inline unsigned long
__find_buddy_index(unsigned long page_idx, unsigned int order)
{
@@ -190,21 +190,17 @@ __find_combined_index(unsigned long page_idx, unsigned int order)
return (page_idx & ~(1 << order));
}
-/*
- * Linux<75>ں˽<DABA><CBBD><EFBFBD><EFBFBD>ҳ<EFBFBD><D2B3>order<65><72>¼<EFBFBD>ڵڶ<DAB5><DAB6><EFBFBD>ҳ<EFBFBD><D2B3><EFBFBD>prevָ<76><D6B8><EFBFBD><EFBFBD>
- * <20><>ϵͳ<CFB5><CDB3><EFBFBD><EFBFBD><EFBFBD>ҳ<EFBFBD><D2B3>order<65><72>¼<EFBFBD><C2BC><EFBFBD>׸<EFBFBD>ҳ<EFBFBD><D2B3><EFBFBD>page->order<65><72><EFBFBD><EFBFBD>
- * */
+
static inline unsigned long esCompound_order(struct esPage_s *page)
{
if (!esPageHead(page))
- return 0; //<2F><>ҳ
- //return (unsigned long)page[1].lru.prev;
+ return 0;
+
return page->order;
}
static inline void esSet_compound_order(struct esPage_s *page, unsigned long order)
{
- //page[1].lru.prev = (void *)order;
page->order = order;
}
diff --git a/drivers/memory/eswin/es_buddy/Kconfig b/drivers/memory/eswin/es_buddy/Kconfig
new file mode 100644
index 000000000000..94d755f4ba86
--- /dev/null
+++ b/drivers/memory/eswin/es_buddy/Kconfig
@@ -0,0 +1,6 @@
+# SPDX-License-Identifier: GPL-2.0
+
+config ESWIN_BUDDY
+ tristate "ESWIN buddy allocator for reserved memory"
+ help
+ buddy allocator api.
diff --git a/drivers/memory/eswin/es_buddy/Makefile b/drivers/memory/eswin/es_buddy/Makefile
new file mode 100644
index 000000000000..4115fd0901d7
--- /dev/null
+++ b/drivers/memory/eswin/es_buddy/Makefile
@@ -0,0 +1,3 @@
+# SPDX-License-Identifier: GPL-2.0
+obj-$(CONFIG_ESWIN_BUDDY) += es_buddy.o buddy_allocator/buddy.o
+
diff --git a/drivers/memory/eswin/es_buddy/buddy_allocator/buddy.c b/drivers/memory/eswin/es_buddy/buddy_allocator/buddy.c
new file mode 100644
index 000000000000..56f084d0fc46
--- /dev/null
+++ b/drivers/memory/eswin/es_buddy/buddy_allocator/buddy.c
@@ -0,0 +1,222 @@
+// SPDX-License-Identifier: GPL-2.0-only
+/*
+ * Internal APIs for ESWIN buddy allocator
+ *
+ * Copyright 2024 Beijing ESWIN Computing Technology Co., Ltd.
+ * Authors:
+ * LinMin<linmin@eswincomputing.com>
+ *
+ */
+
+#include "../../buddy.h"
+
+#define SIZE_4GB 0x100000000
+static int is_4G_boundary_page(struct mem_block *memblock, unsigned long page_idx)
+{
+ struct page *kpage;
+
+ kpage = memblock->kPageStart + page_idx;
+ if (!((page_to_phys(kpage) + BUDDY_PAGE_SIZE) % SIZE_4GB))
+ return 1;
+
+ return 0;
+}
+
+void buddy_system_init(struct mem_block *memblock,
+ struct esPage_s *start_page,
+ unsigned long start_addr,
+ unsigned long page_num)
+{
+ unsigned long i;
+ struct esPage_s *page = NULL;
+ struct es_free_area_s *area = NULL;
+ // init memory zone
+ struct mem_zone *zone = &memblock->zone;
+ zone->page_num = page_num;
+ zone->page_size = BUDDY_PAGE_SIZE;
+ zone->first_page = start_page;
+ zone->start_addr = start_addr;
+ zone->end_addr = start_addr + page_num * BUDDY_PAGE_SIZE;
+ // TODO: init zone->lock
+ #ifdef RUN_IN_KERNEL
+ buddy_spin_lock_init(&zone->lock);
+ #endif
+ // init each area
+ for (i = 0; i < BUDDY_MAX_ORDER; i++)
+ {
+ area = zone->free_area + i;
+ INIT_LIST_HEAD(&area->free_list);
+ area->nr_free = 0;
+ }
+ memset(start_page, 0, page_num * sizeof(struct esPage_s));
+ // init and free each page
+ for (i = 0; i < page_num; i++)
+ {
+ page = zone->first_page + i;
+ INIT_LIST_HEAD(&page->lru);
+ /* Reserve 4kB at (4GB-4k) alignment address boundary. This is a workaround for g2d.
+ The g2d hardware has problem with accessing the 4GB alignment address boundray,
+ such as the address at 4GB, 8GB, 12GB and 16GB.
+ */
+ if (is_4G_boundary_page(memblock, i)) {
+ memblock->page_num--;
+ continue;
+ }
+
+ buddy_free_pages(zone, page);
+ }
+}
+
+static void prepare_compound_pages(struct esPage_s *page, unsigned long order)
+{
+ unsigned long i;
+ unsigned long nr_pages = (1UL<<order);
+
+ esSet_compound_order(page, order);
+ __esSetPageHead(page);
+ for(i = 1; i < nr_pages; i++)
+ {
+ struct esPage_s *p = page + i;
+ __SetPageTail(p);
+ p->first_page = page;
+ }
+}
+
+static void expand(struct mem_zone *zone, struct esPage_s *page,
+ unsigned long low_order, unsigned long high_order,
+ struct es_free_area_s *area)
+{
+ unsigned long size = (1U << high_order);
+ while (high_order > low_order)
+ {
+ area--;
+ high_order--;
+ size >>= 1;
+ list_add(&page[size].lru, &area->free_list);
+ area->nr_free++;
+ // set page order
+ set_page_order_buddy(&page[size], high_order);
+ }
+}
+
+static struct esPage_s *__alloc_page(unsigned long order,
+ struct mem_zone *zone)
+{
+ struct esPage_s *page = NULL;
+ struct es_free_area_s *area = NULL;
+ unsigned long current_order = 0;
+
+ for (current_order = order;
+ current_order < BUDDY_MAX_ORDER; current_order++)
+ {
+ area = zone->free_area + current_order;
+ if (list_empty(&area->free_list)) {
+ continue;
+ }
+ // remove closest size page
+ page = list_entry(area->free_list.next, struct esPage_s, lru);
+ list_del(&page->lru);
+ rmv_page_order_buddy(page);
+ area->nr_free--;
+ // expand to lower order
+ expand(zone, page, order, current_order, area);
+ // compound page
+ if (order > 0)
+ prepare_compound_pages(page, order);
+ else // single page
+ page->order = 0;
+ return page;
+ }
+ return NULL;
+}
+
+struct esPage_s *buddy_get_pages(struct mem_zone *zone,
+ unsigned long order)
+{
+ struct esPage_s *page = NULL;
+
+ if (order >= BUDDY_MAX_ORDER)
+ {
+ BUDDY_BUG(__FILE__, __LINE__);
+ return NULL;
+ }
+ //TODO: lock zone->lock
+ buddy_spin_lock(&zone->lock);
+ page = __alloc_page(order, zone);
+ //TODO: unlock zone->lock
+ buddy_spin_unlock(&zone->lock);
+ return page;
+}
+
+static int destroy_compound_pages(struct esPage_s *page, unsigned long order)
+{
+ int bad = 0;
+ unsigned long i;
+ unsigned long nr_pages = (1UL<<order);
+
+ __esClearPageHead(page);
+ for(i = 1; i < nr_pages; i++)
+ {
+ struct esPage_s *p = page + i;
+ if( !esPageTail(p) || p->first_page != page )
+ {
+ bad++;
+ BUDDY_BUG(__FILE__, __LINE__);
+ }
+ __ClearPageTail(p);
+ }
+ return bad;
+}
+
+#define PageCompound(page) \
+ (page->flags & ((1UL<<enPG_head)|(1UL<<enPG_tail)))
+
+#define page_is_buddy(page,order) \
+ (esPageBuddy(page) && (page->order == order))
+
+void buddy_free_pages(struct mem_zone *zone,
+ struct esPage_s *page)
+{
+ unsigned long order = esCompound_order(page);
+ unsigned long buddy_idx = 0, combinded_idx = 0;
+ unsigned long page_idx = page - zone->first_page;
+ //TODO: lock zone->lock
+ buddy_spin_lock(&zone->lock);
+ if (PageCompound(page))
+ if (destroy_compound_pages(page, order))
+ BUDDY_BUG(__FILE__, __LINE__);
+
+ while (order < BUDDY_MAX_ORDER-1)
+ {
+ struct esPage_s *buddy;
+ // find and delete buddy to combine
+ buddy_idx = __find_buddy_index(page_idx, order);
+ buddy = page + (buddy_idx - page_idx);
+ if (!page_is_buddy(buddy, order))
+ break;
+ list_del(&buddy->lru);
+ zone->free_area[order].nr_free--;
+ // remove buddy's flag and order
+ rmv_page_order_buddy(buddy);
+ // update page and page_idx after combined
+ combinded_idx = __find_combined_index(page_idx, order);
+ page = page + (combinded_idx - page_idx);
+ page_idx = combinded_idx;
+ order++;
+ }
+ set_page_order_buddy(page, order);
+ list_add(&page->lru, &zone->free_area[order].free_list);
+ zone->free_area[order].nr_free++;
+ //TODO: unlock zone->lock
+ buddy_spin_unlock(&zone->lock);
+}
+
+unsigned long buddy_num_free_page(struct mem_zone *zone)
+{
+ unsigned long i, ret;
+ for (i = 0, ret = 0; i < BUDDY_MAX_ORDER; i++)
+ {
+ ret += zone->free_area[i].nr_free * (1UL<<i);
+ }
+ return ret;
+}
diff --git a/drivers/memory/eswin/es_buddy/es_buddy.c b/drivers/memory/eswin/es_buddy/es_buddy.c
new file mode 100644
index 000000000000..bc05cf473af6
--- /dev/null
+++ b/drivers/memory/eswin/es_buddy/es_buddy.c
@@ -0,0 +1,186 @@
+/*
+ * ESWIN buddy allocator.
+ * eswin_rsvmem initializes the reserved memory in dst that has compatible = "eswin-reserve-memory" and
+ * no-map property. Each of these memory region will be treated as one memory block and managed by eswin
+ * buddy system. Users can allocate/frree pages from/to these memory blocks via es_alloc_pages/es_free_pages.
+ *
+ * Copyright 2024 Beijing ESWIN Computing Technology Co., Ltd.
+ * Authors:
+ * LinMin<linmin@eswincomputing.com>
+ *
+ */
+
+#define pr_fmt(fmt) "eswin_buddy: " fmt
+
+#include <linux/memblock.h>
+#include <linux/err.h>
+#include <linux/mm.h>
+#include <linux/module.h>
+#include <linux/sizes.h>
+#include <linux/slab.h>
+#include <linux/log2.h>
+#include <linux/highmem.h>
+#include <linux/io.h>
+#include <linux/kmemleak.h>
+
+#include "../eswin_memblock.h"
+#include "es_buddy.h"
+
+
+static void es_buddy_system_init(struct mem_block *memblock,
+ unsigned long start_addr,
+ unsigned long page_num)
+{
+ struct esPage_s *start_page = memblock->esPagesStart;
+
+ es_spin_lock_init(&memblock->esLock);
+ buddy_system_init(memblock, start_page, start_addr, page_num);
+}
+
+struct page *es_alloc_pages(struct mem_block *memblock,
+ unsigned long order)
+{
+ struct esPage_s *page;
+ struct page *kpage;
+ struct mem_zone *zone = &memblock->zone;
+ unsigned long page_idx;
+
+ es_spin_lock(&memblock->esLock);
+ page = buddy_get_pages(zone, order);
+ if (NULL == page) {
+ es_spin_unlock(&memblock->esLock);
+ return NULL;
+ }
+
+ page_idx = page - zone->first_page;
+ kpage = memblock->kPageStart + page_idx;
+
+ if (order > 0) {
+ __SetPageHead(kpage);
+ set_compound_order(kpage, order);
+ }
+ es_spin_unlock(&memblock->esLock);
+
+ buddy_print("%s:input order=%ld, esCompound_order(page)=%ld, kCompound_order(kpage)=%d, page_size(kpage)=0x%lx, phys_addr=0x%llx\n",
+ __func__, order, esCompound_order(page), compound_order(kpage), page_size(kpage), page_to_phys(kpage));
+ return kpage;
+}
+EXPORT_SYMBOL(es_alloc_pages);
+
+void es_free_pages(struct mem_block *memblock,
+ struct page *kpage)
+{
+ struct mem_zone *zone = &memblock->zone;
+ unsigned long page_idx = kpage - memblock->kPageStart;
+ struct esPage_s *page = zone->first_page + page_idx;
+ unsigned long order = esCompound_order(page);
+
+ buddy_print("%s:esCompound_order(page)=%ld, kCompound_order(kpage)=%d, page_idx=0x%lx, page_size(kpage)=0x%lx, phys_addr=0x%llx\n",
+ __func__, esCompound_order(page), compound_order(kpage), page_idx, page_size(kpage), page_to_phys(kpage));
+ es_spin_lock(&memblock->esLock);
+ buddy_free_pages(zone, page);
+
+ if (order > 0) {
+ ClearPageHead(kpage);
+ }
+ es_spin_unlock(&memblock->esLock);
+
+}
+EXPORT_SYMBOL(es_free_pages);
+
+unsigned long es_num_free_pages(struct mem_block *memblock)
+{
+ struct mem_zone *zone = &memblock->zone;
+
+ return buddy_num_free_page(zone);
+}
+EXPORT_SYMBOL(es_num_free_pages);
+
+void *es_page_to_virt(struct mem_zone *zone,
+ struct esPage_s *page)
+{
+ unsigned long page_idx = 0;
+ unsigned long address = 0;
+
+ page_idx = page - zone->first_page;
+ address = zone->start_addr + page_idx * BUDDY_PAGE_SIZE;
+
+ return (void *)address;
+}
+
+struct esPage_s *es_virt_to_page(struct mem_zone *zone, void *ptr)
+{
+ unsigned long page_idx = 0;
+ struct esPage_s *page = NULL;
+ unsigned long address = (unsigned long)ptr;
+
+ if((address<zone->start_addr)||(address>zone->end_addr))
+ {
+ buddy_print("start_addr=0x%lx, end_addr=0x%lx, address=0x%lx\n",
+ zone->start_addr, zone->end_addr, address);
+ BUDDY_BUG(__FILE__, __LINE__);
+ return NULL;
+ }
+ page_idx = (address - zone->start_addr)>>BUDDY_PAGE_SHIFT;
+
+ page = zone->first_page + page_idx;
+ return page;
+}
+
+static int do_rsvmem_buddy_init(struct mem_block *memblock, void *data)
+{
+ int pages_size;
+
+ pr_debug("eswin buddy init for %s\n", memblock->name);
+ /* alloc esPage_s for all the pages to manage the pages*/
+ pages_size = memblock->page_num * sizeof(struct esPage_s);
+ memblock->esPagesStart = (struct esPage_s*)vmalloc(pages_size);
+ if (!memblock->esPagesStart) {
+ pr_err("%s:%d, failed to buddy init for %s\n",
+ __func__, __LINE__, memblock->name);
+ return -ENOMEM;
+ }
+ es_buddy_system_init(memblock, 0, memblock->page_num);
+
+ return 0;
+}
+static int eswin_rsvmem_buddy_init(void)
+{
+ int ret = 0;
+
+ ret = eswin_rsvmem_for_each_block(do_rsvmem_buddy_init, NULL);
+
+ return ret;
+}
+
+static int do_rsvmem_buddy_uninit(struct mem_block *memblock, void *data)
+{
+ unsigned long numFreePages = 0;
+
+ if (NULL == memblock->esPagesStart)
+ return 0;
+
+ numFreePages = es_num_free_pages(memblock);
+ pr_debug("%s: free mem=0x%lx\n",
+ memblock->name, numFreePages<<PAGE_SHIFT);
+ if (numFreePages == memblock->zone.page_num) {
+ vfree((void*)memblock->esPagesStart);
+ }
+ else {
+ pr_err("%s: %ld outof %ld pages still in use, skip destroy memblock!\n",
+ memblock->name, numFreePages, memblock->zone.page_num);
+
+ return -EBUSY;
+ }
+
+ return 0;
+}
+
+static void __exit eswin_rsvmem_buddy_uninit(void)
+{
+ eswin_rsvmem_for_each_block(do_rsvmem_buddy_uninit, NULL);
+}
+
+subsys_initcall(eswin_rsvmem_buddy_init);
+module_exit(eswin_rsvmem_buddy_uninit);
+MODULE_LICENSE("GPL v2");
\ No newline at end of file
diff --git a/drivers/memory/eswin/es_buddy/es_buddy.h b/drivers/memory/eswin/es_buddy/es_buddy.h
new file mode 100644
index 000000000000..7861a6cbcab8
--- /dev/null
+++ b/drivers/memory/eswin/es_buddy/es_buddy.h
@@ -0,0 +1,52 @@
+// SPDX-License-Identifier: GPL-2.0-only
+/*
+ * Header file of ESWIN buddy allocator
+ *
+ * Copyright 2024 Beijing ESWIN Computing Technology Co., Ltd.
+ * Authors:
+ * LinMin<linmin@eswincomputing.com>
+ *
+ */
+
+#ifndef __ESWIN_BUDDY_H__
+#define __ESWIN_BUDDY_H__
+
+#include <linux/init.h>
+#include <linux/types.h>
+#include <linux/version.h>
+#if LINUX_VERSION_CODE >= KERNEL_VERSION(6,6,0)
+#include <linux/mm_types.h>
+#include <asm/atomic.h>
+#endif
+#include <linux/numa.h>
+#include "../buddy.h"
+
+#if LINUX_VERSION_CODE >= KERNEL_VERSION(6,6,0)
+static inline void folio_set_order(struct folio *folio, unsigned int order)
+{
+ if (WARN_ON_ONCE(!order || !folio_test_large(folio)))
+ return;
+
+ folio->_flags_1 = (folio->_flags_1 & ~0xffUL) | order;
+#ifdef CONFIG_64BIT
+ folio->_folio_nr_pages = 1U << order;
+#endif
+}
+
+static inline void prep_compound_head(struct page *page, unsigned int order)
+{
+ struct folio *folio = (struct folio *)page;
+
+ folio_set_order(folio, order);
+ atomic_set(&folio->_entire_mapcount, -1);
+ atomic_set(&folio->_nr_pages_mapped, 0);
+ atomic_set(&folio->_pincount, 0);
+}
+#define set_compound_order(kpage, order) prep_compound_head(kpage, order)
+#endif
+
+extern struct page *es_alloc_pages(struct mem_block *memblock, unsigned long order);
+extern void es_free_pages(struct mem_block *memblock, struct page *kpage);
+extern unsigned long es_num_free_pages(struct mem_block *memblock);
+
+#endif
diff --git a/drivers/memory/eswin/es_mmz_vb/Kconfig b/drivers/memory/eswin/es_mmz_vb/Kconfig
new file mode 100644
index 000000000000..e290be281abe
--- /dev/null
+++ b/drivers/memory/eswin/es_mmz_vb/Kconfig
@@ -0,0 +1,6 @@
+# SPDX-License-Identifier: GPL-2.0
+
+config ESWIN_MMZ_VB
+ tristate "ESWIN MMZ reserved memory VB"
+ help
+ ESWIN MMZ reserved memory VB device.
diff --git a/drivers/memory/eswin/es_mmz_vb/Makefile b/drivers/memory/eswin/es_mmz_vb/Makefile
new file mode 100644
index 000000000000..fe2bfe6291b1
--- /dev/null
+++ b/drivers/memory/eswin/es_mmz_vb/Makefile
@@ -0,0 +1,5 @@
+# SPDX-License-Identifier: GPL-2.0
+obj-$(CONFIG_ESWIN_MMZ_VB) += mmz_vb.o
+
+
+
diff --git a/drivers/memory/eswin/es_mmz_vb/include/linux/mmz_vb.h b/drivers/memory/eswin/es_mmz_vb/include/linux/mmz_vb.h
new file mode 100644
index 000000000000..e167311a238d
--- /dev/null
+++ b/drivers/memory/eswin/es_mmz_vb/include/linux/mmz_vb.h
@@ -0,0 +1,58 @@
+// SPDX-License-Identifier: GPL-2.0-only
+/*
+ * Copyright 2024 Beijing ESWIN Computing Technology Co., Ltd.
+ * Authors:
+ * LinMin<linmin@eswincomputing.com>
+ *
+ */
+
+#ifndef __MMZ_VB_H__
+#define __MMZ_VB_H__
+
+#include <linux/types.h>
+#include "../../../eswin_memblock.h"
+#include "../../../es_buddy/es_buddy.h"
+
+
+/**
+ * select whether the block will be memset to 0 while exporting it as dmabuf.
+ * 1: memset
+ * 0: do NOT memset
+*/
+// #define MMZ_VB_DMABUF_MEMSET 1
+
+#define MMZ_VB_VALID_FD_FLAGS (O_CLOEXEC | O_ACCMODE)
+
+#define VB_K_POOL_MAX_ID INT_MAX
+
+/* one block info organized in kernel */
+typedef struct esVB_K_BLOCK_INFO_S {
+ struct page *cma_pages;
+ struct sg_table sg_table; // for buddy allocator
+ struct esVB_K_POOL_INFO_S *pool;
+ int nr;
+}VB_K_BLOCK_INFO_S;
+
+/* one pool info organized in kernel */
+typedef struct esVB_K_POOL_INFO_S {
+ s32 poolId;
+ struct esVB_POOL_CONFIG_S poolCfg;
+ unsigned long *bitmap; // used for block get/release managment
+ struct esVB_K_BLOCK_INFO_S *blocks; // point to the block array
+ struct esVB_K_MMZ_S *partitions; // poiont to the partitions
+ struct hlist_node node;
+ spinlock_t lock;
+ enum esVB_UID_E enVbUid;
+ unsigned long flag;
+}VB_K_POOL_INFO_S;
+
+/* MMZs info in kernel */
+typedef struct esVB_K_MMZ_S {
+ u32 partCnt;
+ // pool is found via idr api
+ struct idr pool_idr;
+ struct rw_semaphore idr_lock;
+ struct mem_block *mem_blocks[ES_VB_MAX_MMZs];
+}VB_K_MMZ_S;
+
+#endif
diff --git a/drivers/memory/eswin/es_mmz_vb/include/linux/mmz_vb_type.h b/drivers/memory/eswin/es_mmz_vb/include/linux/mmz_vb_type.h
new file mode 100644
index 000000000000..8adecbbc4d6b
--- /dev/null
+++ b/drivers/memory/eswin/es_mmz_vb/include/linux/mmz_vb_type.h
@@ -0,0 +1,17 @@
+// SPDX-License-Identifier: GPL-2.0-only
+/*
+ * Copyright 2024 Beijing ESWIN Computing Technology Co., Ltd.
+ * Authors:
+ * LinMin<linmin@eswincomputing.com>
+ *
+ */
+
+#ifndef __MMZ_VB_TYPE_H__
+#define __MMZ_VB_TYPE_H__
+
+typedef unsigned int ES_U32;
+typedef unsigned long long ES_U64;
+typedef char ES_CHAR;
+typedef ES_U32 VB_POOL;
+
+#endif
diff --git a/drivers/memory/eswin/es_mmz_vb/mmz_vb.c b/drivers/memory/eswin/es_mmz_vb/mmz_vb.c
new file mode 100644
index 000000000000..2c2f829817ad
--- /dev/null
+++ b/drivers/memory/eswin/es_mmz_vb/mmz_vb.c
@@ -0,0 +1,2375 @@
+// SPDX-License-Identifier: GPL-2.0-only
+/*
+ * ESWIN MMZ VB driver. MMZ VB stands for Media Memory Zone Video Buffer.
+ *
+ * Copyright 2024 Beijing ESWIN Computing Technology Co., Ltd.
+ * Authors:
+ * LinMin<linmin@eswincomputing.com>
+ *
+ */
+
+#include <linux/dma-buf.h>
+#include <linux/dma-heap.h>
+#include <linux/dma-map-ops.h>
+#include <linux/highmem.h>
+#include <linux/bitfield.h>
+#include <linux/module.h>
+#include <linux/moduleparam.h>
+#include <linux/types.h>
+#include <linux/string.h>
+#include <linux/miscdevice.h>
+#include <linux/fs.h>
+#include <linux/init.h>
+#include <linux/slab.h>
+#include <asm/uaccess.h>
+#include <linux/dmaengine.h>
+#include <linux/uaccess.h> /* For copy_to_user/put_user/... */
+#include <linux/of_address.h>
+#include <crypto/hash.h>
+#include <linux/delay.h>
+#include <linux/hashtable.h>
+#include <linux/es_proc.h>
+#include <linux/dmabuf-heap-import-helper.h>
+#include "include/linux/mmz_vb_type.h" /*must include before es_vb_user.h*/
+#include <uapi/linux/es_vb_user.h>
+#include <uapi/linux/mmz_vb.h>
+#include "include/linux/mmz_vb.h"
+
+MODULE_IMPORT_NS(DMA_BUF);
+
+#define DRIVER_NAME "mmz_vb"
+#define MMZ_VB_DMABUF_NAME "mmz_vb_dmabuf"
+#define MMZ_VB_DMABUF_SPLITTED_NAME "mmz_vb_dmabuf_splitted"
+
+#define vb_fmt(fmt) "[%s-MMZ_VB]: " fmt
+#define info_fmt(fmt) vb_fmt("%s[%d]: " fmt), "INFO", \
+ __func__, __LINE__
+#define dbg_fmt(fmt) vb_fmt("%s[%d]: " fmt), "DEBUG", \
+ __func__, __LINE__
+#define err_fmt(fmt) vb_fmt("%s[%d]: " fmt), "ERROR", \
+ __func__, __LINE__
+
+#define vb_info(fmt, args...) \
+ do { \
+ printk(KERN_INFO info_fmt(fmt), ##args); \
+ } while (0)
+
+#define vb_debug(fmt, args...) \
+ do { \
+ printk(KERN_DEBUG dbg_fmt(fmt), ##args); \
+ } while (0)
+#define vb_err(fmt, args...) \
+ do { \
+ printk(KERN_ERR err_fmt(fmt), ##args); \
+ } while (0)
+
+static struct device *mmz_vb_dev;
+static struct mmz_vb_priv *g_mmz_vb_priv = NULL;
+
+struct mmz_vb_priv {
+ struct device *dev;
+ struct esVB_K_MMZ_S partitions;
+ atomic_t allocBlkcnt; /*total block allocated*/
+ struct hlist_head ht[VB_UID_MAX][ES_VB_MAX_MOD_POOL];
+ struct rw_semaphore pool_lock[VB_UID_MAX];
+ unsigned long cfg_flag[VB_UID_MAX]; /*flag for pVbConfig*/
+ struct esVB_CONFIG_S *pVbConfig[VB_UID_MAX];
+ struct mutex cfg_lock[VB_UID_MAX];
+};
+
+#define do_vb_pool_size(pPool) (pPool->poolCfg.blkCnt * pPool->poolCfg.blkSize)
+static int vb_find_pool_by_id_unlock(VB_POOL poolId, struct esVB_K_POOL_INFO_S **ppPool);
+static int vb_find_pool_by_id(VB_POOL poolId, struct esVB_K_POOL_INFO_S **ppPool);
+static int vb_pool_size(VB_POOL poolId, u64 *pPoolSize);
+// static int vb_flush_pool(struct esVB_FLUSH_POOL_CMD_S *flushPoolCmd);
+static int vb_get_block(struct esVB_GET_BLOCK_CMD_S *getBlkCmd, struct esVB_K_BLOCK_INFO_S **ppBlk);
+static void vb_release_block(struct esVB_K_BLOCK_INFO_S *pBlk);
+static int vb_pool_get_free_block_cnt_unlock(struct esVB_K_POOL_INFO_S *pool);
+static int vb_is_splitted_blk(int fd, bool *isSplittedBlk);
+static int vb_blk_to_pool(struct esVB_BLOCK_TO_POOL_CMD_S *blkToPoolCmd);
+static int vb_get_blk_offset(struct esVB_GET_BLOCKOFFSET_CMD_S *getBlkOffsetCmd);
+static int vb_split_dmabuf(struct esVB_SPLIT_DMABUF_CMD_S *splitDmabufCmd);
+static int vb_get_dmabuf_refcnt(struct esVB_DMABUF_REFCOUNT_CMD_S *getDmabufRefCntCmd);
+static int vb_retrieve_mem_node(struct esVB_RETRIEVE_MEM_NODE_CMD_S *retrieveMemNodeCmd);
+static int vb_get_dmabuf_size(struct esVB_DMABUF_SIZE_CMD_S *getDmabufSizeCmd);
+static int mmz_vb_pool_exit(void);
+static int mmz_vb_init_memory_region(void);
+
+/**
+ * vb dmabuf releated struct
+ *
+ */
+struct mmz_vb_buffer {
+ struct esVB_K_BLOCK_INFO_S *pBlk;
+ struct list_head attachments;
+ struct mutex lock;
+ unsigned long len;
+ struct sg_table *table; // for buddy allocator
+ struct page **pages;
+ int vmap_cnt;
+ void *vaddr;
+};
+
+struct mmz_vb_attachment {
+ struct device *dev;
+ struct sg_table *table;
+ struct list_head list;
+ bool mapped;
+};
+
+static struct sg_table *dup_sg_table(struct sg_table *table)
+{
+ struct sg_table *new_table;
+ int ret, i;
+ struct scatterlist *sg, *new_sg;
+
+ new_table = kzalloc(sizeof(*new_table), GFP_KERNEL);
+ if (!new_table)
+ return ERR_PTR(-ENOMEM);
+
+ ret = sg_alloc_table(new_table, table->orig_nents, GFP_KERNEL);
+ if (ret) {
+ kfree(new_table);
+ return ERR_PTR(-ENOMEM);
+ }
+
+ new_sg = new_table->sgl;
+ for_each_sgtable_sg(table, sg, i) {
+ sg_set_page(new_sg, sg_page(sg), sg->length, sg->offset);
+ new_sg = sg_next(new_sg);
+ }
+
+ return new_table;
+}
+
+static int mmz_vb_attach(struct dma_buf *dmabuf,
+ struct dma_buf_attachment *attachment)
+{
+ struct mmz_vb_buffer *buffer = dmabuf->priv;
+ struct mmz_vb_attachment *a;
+ struct sg_table *table;
+
+ a = kzalloc(sizeof(*a), GFP_KERNEL);
+ if (!a)
+ return -ENOMEM;
+
+ table = dup_sg_table(buffer->table);
+ if (IS_ERR(table)) {
+ kfree(a);
+ return -ENOMEM;
+ }
+
+ a->table = table;
+ a->dev = attachment->dev;
+ INIT_LIST_HEAD(&a->list);
+ a->mapped = false;
+
+ attachment->priv = a;
+
+ mutex_lock(&buffer->lock);
+ list_add(&a->list, &buffer->attachments);
+ mutex_unlock(&buffer->lock);
+
+ return 0;
+}
+
+static void mmz_vb_detach(struct dma_buf *dmabuf,
+ struct dma_buf_attachment *attachment)
+{
+ struct mmz_vb_buffer *buffer = dmabuf->priv;
+ struct mmz_vb_attachment *a = attachment->priv;
+
+ mutex_lock(&buffer->lock);
+ list_del(&a->list);
+ mutex_unlock(&buffer->lock);
+
+ sg_free_table(a->table);
+ kfree(a->table);
+ kfree(a);
+}
+
+static struct sg_table *mmz_vb_map_dma_buf(struct dma_buf_attachment *attachment,
+ enum dma_data_direction direction)
+{
+ struct mmz_vb_attachment *a = attachment->priv;
+ struct sg_table *table =a->table;
+ int ret;
+
+ /* Skipt cache sync, since it takes a lot of time when import to device.
+ * It's the user's responsibility for guaranteeing the cache coherency by
+ flusing cache explicitly before importing to device.
+ */
+ ret = dma_map_sgtable(attachment->dev, table, direction, DMA_ATTR_SKIP_CPU_SYNC);
+
+ if (ret)
+ return ERR_PTR(-ENOMEM);
+ a->mapped = true;
+ return table;
+}
+
+static void mmz_vb_unmap_dma_buf(struct dma_buf_attachment *attachment,
+ struct sg_table *table,
+ enum dma_data_direction direction)
+{
+ struct mmz_vb_attachment *a = attachment->priv;
+
+ a->mapped = false;
+
+ /* Skipt cache sync, since it takes a lot of time when unmap from device.
+ * It's the user's responsibility for guaranteeing the cache coherency after
+ the device has done processing the data.(For example, CPU do NOT read untill
+ the device has done)
+ */
+ dma_unmap_sgtable(attachment->dev, table, direction, DMA_ATTR_SKIP_CPU_SYNC);
+}
+
+static int mmz_vb_dma_buf_begin_cpu_access(struct dma_buf *dmabuf,
+ enum dma_data_direction direction)
+{
+ struct mmz_vb_buffer *buffer = dmabuf->priv;
+ struct sg_table *table = buffer->table;
+ struct scatterlist *sg;
+ int i;
+
+ mutex_lock(&buffer->lock);
+
+ if (buffer->vmap_cnt)
+ invalidate_kernel_vmap_range(buffer->vaddr, buffer->len);
+
+ /* Since the cache sync was skipped when mmz_vb_map_dma_buf/mmz_vb_unmap_dma_buf,
+ So force cache sync here when user call ES_SYS_MemFlushCache, eventhough there
+ is no device attached to this dmabuf.
+ */
+ #ifndef QEMU_DEBUG
+ for_each_sg(table->sgl, sg, table->orig_nents, i)
+ arch_sync_dma_for_cpu(sg_phys(sg), sg->length, direction);
+ #endif
+ mutex_unlock(&buffer->lock);
+
+ return 0;
+}
+
+static int mmz_vb_dma_buf_end_cpu_access(struct dma_buf *dmabuf,
+ enum dma_data_direction direction)
+{
+ struct mmz_vb_buffer *buffer = dmabuf->priv;
+ struct sg_table *table = buffer->table;
+ struct scatterlist *sg;
+ int i;
+
+ mutex_lock(&buffer->lock);
+
+ if (buffer->vmap_cnt)
+ flush_kernel_vmap_range(buffer->vaddr, buffer->len);
+
+ /* Since the cache sync was skipped while mmz_vb_map_dma_buf/mmz_vb_unmap_dma_buf,
+ So force cache sync here when user call ES_SYS_MemFlushCache, eventhough there
+ is no device attached to this dmabuf.
+ */
+ #ifndef QEMU_DEBUG
+ for_each_sg(table->sgl, sg, table->orig_nents, i)
+ arch_sync_dma_for_device(sg_phys(sg), sg->length, direction);
+ #endif
+ mutex_unlock(&buffer->lock);
+
+ return 0;
+}
+
+#if 0
+static int mmz_vb_sync_cache_internal(struct dma_buf *dmabuf, enum dma_data_direction direction)
+{
+ struct mmz_vb_buffer *buffer = dmabuf->priv;
+ struct sg_table *table = buffer->table;
+ struct scatterlist *sg;
+ int i;
+
+ for_each_sg(table->sgl, sg, table->orig_nents, i)
+ arch_sync_dma_for_device(sg_phys(sg), sg->length, direction);
+
+
+ return 0;
+}
+#endif
+
+static int mmz_vb_mmap(struct dma_buf *dmabuf, struct vm_area_struct *vma)
+{
+ struct mmz_vb_buffer *buffer = dmabuf->priv;
+ struct sg_table *table = buffer->table;
+ unsigned long addr = vma->vm_start;
+ unsigned long pgoff = vma->vm_pgoff, mapsize = 0;
+ unsigned long size_remaining = vma->vm_end - vma->vm_start;
+ struct scatterlist *sg;
+ struct page *page = NULL;
+ unsigned int nents = 0;
+ int i;
+ int ret;
+
+ if ((vma->vm_flags & (VM_SHARED | VM_MAYSHARE)) == 0)
+ return -EINVAL;
+
+ /* vm_private_data will be used by eswin-ipc-scpu.c.
+ ipc will import this dmabuf to get iova.
+ */
+ vma->vm_private_data = dmabuf;
+
+ /* support mman flag MAP_SHARED_VALIDATE | VM_NORESERVE, used to map uncached memory to user space.
+ The cache needs to be flush first since there might be dirty data in cache.
+ */
+ if (vma->vm_flags & VM_NORESERVE) {
+ vm_flags_clear(vma, VM_NORESERVE);
+ #ifndef QEMU_DEBUG
+ vma->vm_page_prot = pgprot_dmacoherent(vma->vm_page_prot);
+ #endif
+ /* skip sync cache, users should guarantee the cache is clean after done using it in
+ cached mode(i.e, ES_SYS_Mmap(SYS_CACHE_MODE_CACHED))
+ */
+ #if 0
+ pr_debug("%s uncached user memory, flush cache firstly!\n", __func__);
+ if (mmz_vb_sync_cache_internal(dmabuf, DMA_TO_DEVICE)) {
+ vb_err("%s, failed to flush cache!\n",__func__);
+ return -EINVAL;
+ }
+ #endif
+ }
+ pr_debug("%s, size_remaining:0x%lx, pgoff:0x%lx, dmabuf->size:0x%lx, start_phys:0x%llx\n",
+ __func__, size_remaining, pgoff, dmabuf->size, sg_phys(table->sgl));
+ for_each_sg(table->sgl, sg, table->orig_nents, i) {
+ pr_debug("sgl:%d, phys:0x%llx, length:0x%x\n", i, sg_phys(sg), sg->length);
+ if (pgoff >= (sg->length >> PAGE_SHIFT)) {
+ pgoff -= (sg->length >> PAGE_SHIFT);
+ continue;
+ }
+
+ page = sg_page(sg);
+ if (nents == 0) {
+ mapsize = sg->length - (pgoff << PAGE_SHIFT);
+ mapsize = min(size_remaining, mapsize);
+ ret = remap_pfn_range(vma, addr, page_to_pfn(page) + pgoff, mapsize,
+ vma->vm_page_prot);
+ pr_debug("nents:%d, sgl:%d, pgoff:0x%lx, mapsize:0x%lx, phys:0x%llx\n",
+ nents, i, pgoff, mapsize, pfn_to_phys(page_to_pfn(page) + pgoff));
+ }
+ else {
+ mapsize = min((unsigned int)size_remaining, (sg->length));
+ ret = remap_pfn_range(vma, addr, page_to_pfn(page), mapsize,
+ vma->vm_page_prot);
+ pr_debug("nents:%d, sgl:%d, mapsize:0x%lx, phys:0x%llx\n", nents, i, mapsize, page_to_phys(page));
+ }
+ pgoff = 0;
+ nents++;
+
+ if (ret)
+ return ret;
+
+ addr += mapsize;
+ size_remaining -= mapsize;
+ if (size_remaining == 0)
+ return 0;
+ }
+
+ return 0;
+}
+
+static void *mmz_vb_do_vmap(struct dma_buf *dmabuf)
+{
+ struct mmz_vb_buffer *buffer = dmabuf->priv;
+ struct esVB_K_POOL_INFO_S *pool = buffer->pBlk->pool;
+ pgprot_t prot = PAGE_KERNEL;
+ struct sg_table *table = buffer->table;
+ int npages = PAGE_ALIGN(buffer->len) / PAGE_SIZE;
+ struct page **pages = vmalloc(sizeof(struct page *) * npages);
+ struct page **tmp = pages;
+ struct sg_page_iter piter;
+ void *vaddr;
+
+ if (!pages)
+ return ERR_PTR(-ENOMEM);
+
+ for_each_sgtable_page(table, &piter, 0) {
+ WARN_ON(tmp - pages >= npages);
+ *tmp++ = sg_page_iter_page(&piter);
+ }
+
+ /* The property of this dmabuf in kernel space is determined by SYS_CACHE_MODE_E of the pool . */
+ if (pool->poolCfg.enRemapMode == SYS_CACHE_MODE_NOCACHE) {
+ pr_debug("%s uncached kernel buffer!\n", __func__);
+ #ifndef QEMU_DEBUG
+ prot = pgprot_dmacoherent(PAGE_KERNEL);
+ #endif
+ }
+ else {
+ pr_debug("%s cached kernel buffer!\n", __func__);
+ }
+
+ vaddr = vmap(pages, npages, VM_MAP, prot);
+ vfree(pages);
+
+ if (!vaddr)
+ return ERR_PTR(-ENOMEM);
+
+ return vaddr;
+}
+
+static int mmz_vb_vmap(struct dma_buf *dmabuf, struct dma_buf_map *map)
+{
+ struct mmz_vb_buffer *buffer = dmabuf->priv;
+ void *vaddr;
+ int ret = 0;
+
+ mutex_lock(&buffer->lock);
+ if (buffer->vmap_cnt) {
+ buffer->vmap_cnt++;
+ dma_buf_map_set_vaddr(map, buffer->vaddr);
+ goto out;
+ }
+
+ vaddr = mmz_vb_do_vmap(dmabuf);
+ if (IS_ERR(vaddr)) {
+ ret = PTR_ERR(vaddr);
+ goto out;
+ }
+ buffer->vaddr = vaddr;
+ buffer->vmap_cnt++;
+ dma_buf_map_set_vaddr(map, buffer->vaddr);
+out:
+ mutex_unlock(&buffer->lock);
+
+ return ret;
+}
+
+static void mmz_vb_vunmap(struct dma_buf *dmabuf, struct dma_buf_map *map)
+{
+ struct mmz_vb_buffer *buffer = dmabuf->priv;
+
+ mutex_lock(&buffer->lock);
+ if (!--buffer->vmap_cnt) {
+ vunmap(buffer->vaddr);
+ buffer->vaddr = NULL;
+ }
+ mutex_unlock(&buffer->lock);
+ dma_buf_map_clear(map);
+}
+
+static void mmz_vb_dma_buf_release(struct dma_buf *dmabuf)
+{
+ struct mmz_vb_buffer *buffer = dmabuf->priv;
+
+ if (buffer->vmap_cnt > 0) {
+ WARN(1, "%s: buffer still mapped in the kernel\n", __func__);
+ vunmap(buffer->vaddr);
+ buffer->vaddr = NULL;
+ }
+
+ /* release block. In fact, release block to pool */
+ vb_release_block(buffer->pBlk);
+
+ kfree(buffer);
+}
+
+static const struct dma_buf_ops mmz_vb_buf_ops = {
+ .attach = mmz_vb_attach,
+ .detach = mmz_vb_detach,
+ .map_dma_buf = mmz_vb_map_dma_buf,
+ .unmap_dma_buf = mmz_vb_unmap_dma_buf,
+ .begin_cpu_access = mmz_vb_dma_buf_begin_cpu_access,
+ .end_cpu_access = mmz_vb_dma_buf_end_cpu_access,
+ .mmap = mmz_vb_mmap,
+ .vmap = mmz_vb_vmap,
+ .vunmap = mmz_vb_vunmap,
+ .release = mmz_vb_dma_buf_release,
+};
+
+static const unsigned int orders[] = {MAX_ORDER-1, 9, 0};
+#define NUM_ORDERS ARRAY_SIZE(orders)
+
+static struct page *alloc_largest_available(struct mem_block *memblock,
+ unsigned long size,
+ unsigned int max_order)
+{
+ struct page *page;
+ int i;
+
+ for (i = 0; i < NUM_ORDERS; i++) {
+ if (size < (PAGE_SIZE << orders[i]))
+ continue;
+ if (max_order < orders[i])
+ continue;
+
+ page = es_alloc_pages(memblock, orders[i]);
+ if (!page)
+ continue;
+ return page;
+ }
+ return NULL;
+}
+
+static int vb_blk_pages_allocate(struct mem_block *memblock, struct esVB_K_BLOCK_INFO_S *blocks, unsigned long len)
+{
+ unsigned long size_remaining = len;
+ unsigned int max_order = orders[0];
+ struct sg_table *table;
+ struct scatterlist *sg;
+ struct list_head pages;
+ struct page *page, *tmp_page;
+ int i, ret = -ENOMEM;
+
+ INIT_LIST_HEAD(&pages);
+ i = 0;
+ while (size_remaining > 0) {
+ /*
+ * Avoid trying to allocate memory if the process
+ * has been killed by SIGKILL
+ */
+ if (fatal_signal_pending(current)) {
+ ret = -EINTR;
+ goto free_buffer;
+ }
+
+ page = alloc_largest_available(memblock, size_remaining, max_order);
+ if (!page)
+ goto free_buffer;
+
+ list_add_tail(&page->lru, &pages);
+ size_remaining -= page_size(page);
+ max_order = compound_order(page);
+ i++;
+ // pr_debug("page_size(page)=0x%lx, phys_addr=0x%llx, max_order=%d\n",
+ // page_size(page), page_to_phys(page), max_order);
+
+ }
+
+ table = &blocks->sg_table;
+ if (sg_alloc_table(table, i, GFP_KERNEL))
+ goto free_buffer;
+
+ sg = table->sgl;
+ list_for_each_entry_safe(page, tmp_page, &pages, lru) {
+ sg_set_page(sg, page, page_size(page), 0);
+ sg = sg_next(sg);
+ list_del(&page->lru);
+ }
+
+ return 0;
+
+free_buffer:
+ list_for_each_entry_safe(page, tmp_page, &pages, lru)
+ es_free_pages(memblock, page);
+
+
+ return ret;
+}
+
+static void vb_blk_pages_release(struct mem_block *memblock, struct esVB_K_BLOCK_INFO_S *blocks)
+{
+ struct sg_table *table;
+ struct scatterlist *sg;
+ int i;
+
+ table = &blocks->sg_table;
+ for_each_sgtable_sg(table, sg, i) {
+ struct page *page = sg_page(sg);
+ // pr_debug("%s:%d,page_size(page)=0x%lx, phys_addr=0x%llx\n",
+ // __func__, __LINE__, page_size(page), page_to_phys(page));
+ es_free_pages(memblock, page);
+ }
+ sg_free_table(table);
+
+}
+
+static int vb_blk_dmabuf_alloc(struct esVB_GET_BLOCK_CMD_S *getBlkCmd)
+{
+ struct mmz_vb_buffer *buffer;
+ struct esVB_K_BLOCK_INFO_S *pBlk;
+ DEFINE_DMA_BUF_EXPORT_INFO(exp_info);
+ size_t size;
+ struct dma_buf *dmabuf;
+ #ifdef MMZ_VB_DMABUF_MEMSET
+ struct sg_table *table;
+ struct scatterlist *sg;
+ int i;
+ #endif
+ int fd;
+ int ret = -ENOMEM;
+
+ buffer = kzalloc(sizeof(*buffer), GFP_KERNEL);
+ if (!buffer)
+ return -ENOMEM;
+
+ INIT_LIST_HEAD(&buffer->attachments);
+ mutex_init(&buffer->lock);
+
+ ret = vb_get_block(getBlkCmd, &pBlk);
+ /* try to get a required block from pool */
+ if (ret) {
+ vb_err("failed to get block from pool!!!\n");
+ goto free_buffer;
+ }
+
+ size = pBlk->pool->poolCfg.blkSize;
+ buffer->len = size;
+ buffer->table = &pBlk->sg_table;
+ #ifdef MMZ_VB_DMABUF_MEMSET
+ /*TODO: Clear the pages, sg_virt dose not work because vb memory is reserved as no-map!!!!*/
+ #if 0
+ {
+ table = buffer->table;
+ for_each_sg(table->sgl, sg, table->orig_nents, i)
+ memset(sg_virt(sg), 0, sg->length);
+ }
+ #endif
+ #endif
+ buffer->pBlk = pBlk;
+
+ /* create the dmabuf */
+ exp_info.exp_name = MMZ_VB_DMABUF_NAME;
+ exp_info.ops = &mmz_vb_buf_ops;
+ exp_info.size = buffer->len;
+ exp_info.flags = O_RDWR | O_CLOEXEC;
+ exp_info.priv = buffer;
+ dmabuf = dma_buf_export(&exp_info);
+ if (IS_ERR(dmabuf)) {
+ ret = PTR_ERR(dmabuf);
+ goto release_block;
+ }
+
+ fd = dma_buf_fd(dmabuf, MMZ_VB_VALID_FD_FLAGS);
+ if (fd < 0) {
+ dma_buf_put(dmabuf);
+ /* just return, as put will call release and that will free */
+ return fd;
+ }
+
+ getBlkCmd->getBlkResp.fd = fd;
+ getBlkCmd->getBlkResp.actualBlkSize = size;
+ getBlkCmd->getBlkResp.nr = pBlk->nr;
+ return 0;
+
+release_block:
+ vb_release_block(pBlk);
+free_buffer:
+ kfree(buffer);
+
+ return ret;
+}
+
+static int vb_ioctl_get_blk(void __user *user_getBlkCmd)
+{
+ int ret = 0;
+ struct esVB_GET_BLOCK_CMD_S *getBlkCmd;
+
+ getBlkCmd = kzalloc(sizeof(*getBlkCmd), GFP_KERNEL);
+ if (!getBlkCmd) {
+ return -ENOMEM;
+ }
+ if (copy_from_user(getBlkCmd, user_getBlkCmd, sizeof(*getBlkCmd))) {
+ ret = -EFAULT;
+ goto out_free;
+ }
+
+ ret = vb_blk_dmabuf_alloc(getBlkCmd);
+ if (ret) {
+ goto out_free;
+ }
+
+ if (copy_to_user(user_getBlkCmd, getBlkCmd, sizeof(*getBlkCmd)))
+ ret = -EFAULT;
+
+out_free:
+ kfree(getBlkCmd);
+ return ret;
+}
+
+static int vb_ioctl_pool_size(void __user *user_getPoolSizeCmd)
+{
+ int ret = 0;
+ struct esVB_GET_POOLSIZE_CMD_S *getPoolSizeCmd;
+
+ getPoolSizeCmd = kzalloc(sizeof(*getPoolSizeCmd), GFP_KERNEL);
+ if (!getPoolSizeCmd) {
+ return -ENOMEM;
+ }
+ if (copy_from_user(getPoolSizeCmd, user_getPoolSizeCmd, sizeof(*getPoolSizeCmd))) {
+ ret = -EFAULT;
+ goto out_free;
+ }
+
+ ret = vb_pool_size(getPoolSizeCmd->poolId, &getPoolSizeCmd->poolSize);
+ if (ret) {
+ goto out_free;
+ }
+
+ if (copy_to_user(user_getPoolSizeCmd, getPoolSizeCmd, sizeof(*getPoolSizeCmd)))
+ ret = -EFAULT;
+
+out_free:
+ kfree(getPoolSizeCmd);
+ return ret;
+}
+#if 0
+static int vb_ioctl_flush_pool(void __user *user_flushPoolCmd)
+{
+ int ret = 0;
+ struct esVB_FLUSH_POOL_CMD_S *flushPoolCmd;
+
+ flushPoolCmd = kzalloc(sizeof(*flushPoolCmd), GFP_KERNEL);
+ if (!flushPoolCmd) {
+ return -ENOMEM;
+ }
+ if (copy_from_user(flushPoolCmd, user_flushPoolCmd, sizeof(*flushPoolCmd))) {
+ ret = -EFAULT;
+ goto out_free;
+ }
+
+ ret = vb_flush_pool(flushPoolCmd);
+ if (ret) {
+ goto out_free;
+ }
+
+ if (copy_to_user(user_flushPoolCmd, flushPoolCmd, sizeof(*flushPoolCmd)))
+ ret = -EFAULT;
+
+out_free:
+ kfree(flushPoolCmd);
+ return ret;
+}
+#endif
+
+static int vb_ioctl_blk_to_pool(void __user *user_blkToPoolCmd)
+{
+ int ret = 0;
+ struct esVB_BLOCK_TO_POOL_CMD_S *blkToPoolCmd;
+
+ blkToPoolCmd = kzalloc(sizeof(*blkToPoolCmd), GFP_KERNEL);
+ if (!blkToPoolCmd) {
+ return -ENOMEM;
+ }
+ if (copy_from_user(blkToPoolCmd, user_blkToPoolCmd, sizeof(*blkToPoolCmd))) {
+ ret = -EFAULT;
+ goto out_free;
+ }
+
+ ret = vb_blk_to_pool(blkToPoolCmd);
+ if (ret)
+ goto out_free;
+
+ if (copy_to_user(user_blkToPoolCmd, blkToPoolCmd, sizeof(*blkToPoolCmd)))
+ ret = -EFAULT;
+
+out_free:
+ kfree(blkToPoolCmd);
+ return ret;
+}
+
+static int vb_ioctl_get_blk_offset(void __user *user_getBlkOffsetCmd)
+{
+ int ret = 0;
+ struct esVB_GET_BLOCKOFFSET_CMD_S *getBlkOffsetCmd;
+
+ getBlkOffsetCmd = kzalloc(sizeof(*getBlkOffsetCmd), GFP_KERNEL);
+ if (!getBlkOffsetCmd) {
+ return -ENOMEM;
+ }
+ if (copy_from_user(getBlkOffsetCmd, user_getBlkOffsetCmd, sizeof(*getBlkOffsetCmd))) {
+ ret = -EFAULT;
+ goto out_free;
+ }
+
+ ret = vb_get_blk_offset(getBlkOffsetCmd);
+ if (ret) {
+ goto out_free;
+ }
+
+ if (copy_to_user(user_getBlkOffsetCmd, getBlkOffsetCmd, sizeof(*getBlkOffsetCmd)))
+ ret = -EFAULT;
+
+out_free:
+ kfree(getBlkOffsetCmd);
+ return ret;
+}
+
+static int vb_ioctl_split_dmabuf(void __user *user_splitDmabufCmd)
+{
+ int ret = 0;
+ struct esVB_SPLIT_DMABUF_CMD_S *splitDmabufCmd;
+
+ splitDmabufCmd = kzalloc(sizeof(*splitDmabufCmd), GFP_KERNEL);
+ if (!splitDmabufCmd) {
+ return -ENOMEM;
+ }
+ if (copy_from_user(splitDmabufCmd, user_splitDmabufCmd, sizeof(*splitDmabufCmd))) {
+ ret = -EFAULT;
+ goto out_free;
+ }
+
+ ret = vb_split_dmabuf(splitDmabufCmd);
+ if (ret) {
+ goto out_free;
+ }
+
+ if (copy_to_user(user_splitDmabufCmd, splitDmabufCmd, sizeof(*splitDmabufCmd)))
+ ret = -EFAULT;
+
+out_free:
+ kfree(splitDmabufCmd);
+ return ret;
+}
+
+static int vb_ioctl_get_dmabuf_refcnt(void __user *user_getDmabufRefCntCmd)
+{
+ int ret = 0;
+ struct esVB_DMABUF_REFCOUNT_CMD_S *getDmabufRefCntCmd;
+
+ getDmabufRefCntCmd = kzalloc(sizeof(*getDmabufRefCntCmd), GFP_KERNEL);
+ if (!getDmabufRefCntCmd) {
+ return -ENOMEM;
+ }
+ if (copy_from_user(getDmabufRefCntCmd, user_getDmabufRefCntCmd, sizeof(*getDmabufRefCntCmd))) {
+ ret = -EFAULT;
+ goto out_free;
+ }
+
+ ret = vb_get_dmabuf_refcnt(getDmabufRefCntCmd);
+ if (ret)
+ goto out_free;
+
+ if (copy_to_user(user_getDmabufRefCntCmd, getDmabufRefCntCmd, sizeof(*getDmabufRefCntCmd)))
+ ret = -EFAULT;
+
+out_free:
+ kfree(getDmabufRefCntCmd);
+ return ret;
+}
+
+static int vb_ioctl_retrieve_mem_node(void __user *user_retrieveMemNodeCmd)
+{
+ int ret = 0;
+ struct esVB_RETRIEVE_MEM_NODE_CMD_S *retrieveMemNodeCmd;
+
+ retrieveMemNodeCmd = kzalloc(sizeof(*retrieveMemNodeCmd), GFP_KERNEL);
+ if (!retrieveMemNodeCmd) {
+ return -ENOMEM;
+ }
+ if (copy_from_user(retrieveMemNodeCmd, user_retrieveMemNodeCmd, sizeof(*retrieveMemNodeCmd))) {
+ ret = -EFAULT;
+ goto out_free;
+ }
+
+ ret = vb_retrieve_mem_node(retrieveMemNodeCmd);
+ if (ret)
+ goto out_free;
+
+ if (copy_to_user(user_retrieveMemNodeCmd, retrieveMemNodeCmd, sizeof(*retrieveMemNodeCmd)))
+ ret = -EFAULT;
+
+out_free:
+ kfree(retrieveMemNodeCmd);
+ return ret;
+}
+
+static int vb_ioctl_get_dmabuf_size(void __user *user_getDmabufSizeCmd)
+{
+ int ret = 0;
+ struct esVB_DMABUF_SIZE_CMD_S *getDmabufSizeCmd;
+
+ getDmabufSizeCmd = kzalloc(sizeof(*getDmabufSizeCmd), GFP_KERNEL);
+ if (!getDmabufSizeCmd) {
+ return -ENOMEM;
+ }
+ if (copy_from_user(getDmabufSizeCmd, user_getDmabufSizeCmd, sizeof(*getDmabufSizeCmd))) {
+ ret = -EFAULT;
+ goto out_free;
+ }
+
+ ret = vb_get_dmabuf_size(getDmabufSizeCmd);
+ if (ret)
+ goto out_free;
+
+ if (copy_to_user(user_getDmabufSizeCmd, getDmabufSizeCmd, sizeof(*getDmabufSizeCmd)))
+ ret = -EFAULT;
+
+out_free:
+ kfree(getDmabufSizeCmd);
+ return ret;
+}
+
+static int mmz_vb_assign_pool_id(struct esVB_K_POOL_INFO_S *pool)
+{
+ int ret = 0;
+ struct mmz_vb_priv *vb_priv = g_mmz_vb_priv;
+ struct esVB_K_MMZ_S *partitions = &vb_priv->partitions;
+
+ down_write(&partitions->idr_lock);
+ ret = idr_alloc(&partitions->pool_idr, pool, 0, VB_K_POOL_MAX_ID,
+ GFP_KERNEL);
+ if (ret >= 0) {
+ pool->poolId = ret;
+ }
+ up_write(&partitions->idr_lock);
+
+ return ret < 0 ? ret : 0;
+}
+
+static int mmz_vb_remove_pool_id(struct esVB_K_POOL_INFO_S *pool, bool is_lock)
+{
+ int ret = 0;
+ struct mmz_vb_priv *vb_priv = g_mmz_vb_priv;
+ struct esVB_K_MMZ_S *partitions = &vb_priv->partitions;
+
+ if (is_lock) {
+ down_write(&partitions->idr_lock);
+ }
+
+ idr_remove(&partitions->pool_idr, pool->poolId);
+
+ if (is_lock) {
+ up_write(&partitions->idr_lock);
+ }
+ return ret < 0 ? ret : 0;
+}
+
+static int mmz_pool_insert_list(struct esVB_K_POOL_INFO_S *pool, enum esVB_UID_E uid)
+{
+ struct mmz_vb_priv *vb_priv = g_mmz_vb_priv;
+
+ if (uid <= VB_UID_PRIVATE || uid >= VB_UID_MAX) {
+ dev_err(mmz_vb_dev, "%s %d, invalid uid %d\n",__func__,__LINE__, uid);
+ return -EINVAL;
+ }
+ down_write(&vb_priv->pool_lock[uid]);
+ hash_add(vb_priv->ht[uid], &pool->node, pool->poolCfg.blkSize);
+ up_write(&vb_priv->pool_lock[uid]);
+ return 0;
+}
+
+static struct mem_block *vb_get_memblock(const char *memBlkName)
+{
+ struct mmz_vb_priv *vb_priv = g_mmz_vb_priv;
+ struct esVB_K_MMZ_S *partitions = &vb_priv->partitions;
+ struct mem_block *memblock = NULL, *rsvmem_block = NULL;
+ int i;
+
+ for (i = 0; i < partitions->partCnt; i++) {
+ rsvmem_block = partitions->mem_blocks[i];
+ if (!strcmp(memBlkName, rsvmem_block->name)){
+ memblock = rsvmem_block;
+ break;
+ }
+ }
+
+ return memblock;
+}
+
+static int mmz_vb_do_create_pool(struct esVB_POOL_CONFIG_S *pool_cfg,
+ struct esVB_K_POOL_INFO_S **pool_out)
+{
+ int i;
+ int ret = 0;
+ struct esVB_K_POOL_INFO_S *pool;
+ struct esVB_K_BLOCK_INFO_S *blocks;
+ struct mem_block *memblock = NULL;
+ const char *memBlkName = pool_cfg->mmzName;
+ size_t size;
+
+ // 0.find the memblock
+ memblock = vb_get_memblock(memBlkName);
+ if (NULL == memblock) {
+ vb_err("%s NOT found!\n", memBlkName);
+ return -EINVAL;
+ }
+
+ // 1.init pool
+ pool = devm_kzalloc(mmz_vb_dev, sizeof(struct esVB_K_POOL_INFO_S), GFP_KERNEL);
+ if (!pool) {
+ dev_err(mmz_vb_dev, "%s %d, faild to alloc pool cb\n",__func__,__LINE__);
+ ret = -ENOMEM;
+ goto out;
+ }
+
+ pool->blocks = vmalloc(sizeof(struct esVB_K_BLOCK_INFO_S) * pool_cfg->blkCnt);
+ if (!pool->blocks) {
+ dev_err(mmz_vb_dev, "%s %d, faild to alloc blocks cb\n",__func__,__LINE__);
+ ret = -ENOMEM;
+ goto out_free_pool;
+ }
+
+ spin_lock_init(&pool->lock);
+ memcpy(&pool->poolCfg, pool_cfg, sizeof(struct esVB_POOL_CONFIG_S));
+
+ pool->bitmap = bitmap_zalloc(pool_cfg->blkCnt, GFP_KERNEL);
+ if (!pool->bitmap) {
+ dev_err(mmz_vb_dev, "%s %d, faild to alloc bitmap\n",__func__,__LINE__);
+ ret = -ENOMEM;
+ goto out_free_block_arrays;
+ }
+
+ // 2. make blkSize align
+ size = PAGE_ALIGN(pool_cfg->blkSize);
+ /* If len >= 1MB, align len with 2M to improve performance of SMMU */
+ if (size/(PAGE_SIZE << 8)) {
+ size = ALIGN(size, (PAGE_SIZE << 9));
+ }
+ pool_cfg->blkSize = size;
+ pool->poolCfg.blkSize = pool_cfg->blkSize;
+ dev_dbg(mmz_vb_dev, "blkSize(0x%llx) from pool creation is "
+ "aligned to 0x%lx to improve performance.\n",
+ pool_cfg->blkSize, size);
+
+ // 3. alloc pages for blocks
+ for (i = 0; i < pool_cfg->blkCnt; i++) {
+ blocks = &pool->blocks[i];
+ blocks->nr = i;
+ blocks->pool = pool;
+ ret = vb_blk_pages_allocate(memblock, blocks, pool_cfg->blkSize);
+ if (ret) {
+ while (--i >= 0) {
+ vb_blk_pages_release(memblock, &pool->blocks[i]);
+ }
+ dev_err(mmz_vb_dev, "%s %d, faild to alloc block page!\n", __func__,__LINE__);
+ ret = -ENOMEM;
+ goto out_free_bitmap;
+ }
+ }
+ // 4. everthing is ok, add pool to idr
+ ret = mmz_vb_assign_pool_id(pool);
+ if (0 != ret) {
+ dev_err(mmz_vb_dev, "%s %d, faild to assign pool id\n",__func__,__LINE__);
+ ret = -EINVAL;
+ goto out_free_block_pages;
+ }
+ *pool_out = pool;
+ return ret;
+
+out_free_block_pages:
+ for (i = 0; i < pool_cfg->blkCnt; i++) {
+ vb_blk_pages_release(memblock, &pool->blocks[i]);
+ }
+out_free_bitmap:
+ bitmap_free(pool->bitmap);;
+out_free_block_arrays:
+ vfree(pool->blocks);
+out_free_pool:
+ devm_kfree(mmz_vb_dev, pool);
+out:
+ return ret;
+}
+
+static int vb_pool_config_check(struct esVB_POOL_CONFIG_S *pool_cfg)
+{
+ struct mmz_vb_priv *vb_priv = g_mmz_vb_priv;
+ struct mem_block *memblock = NULL;
+ const char *memBlkName = pool_cfg->mmzName;
+ unsigned long numFreePages = 0;
+ u64 req_size;
+
+ if (NULL == vb_priv) {
+ return 0;
+ }
+
+ // find the memblock
+ memblock = vb_get_memblock(memBlkName);
+ if (NULL == memblock) {
+ vb_err("%s NOT found!\n", memBlkName);
+ return -EINVAL;
+ }
+
+ req_size = pool_cfg->blkCnt * PAGE_ALIGN(pool_cfg->blkSize);
+ numFreePages = es_num_free_pages(memblock);
+ if (numFreePages < (req_size >> PAGE_SHIFT)) {
+ dev_err(mmz_vb_dev, "%s %d, (%s)out of memory, request pool size %llu "
+ "free %ld!\n",
+ __func__,__LINE__,
+ memBlkName, req_size, (numFreePages << PAGE_SHIFT));
+ return -ENOMEM;
+ }
+
+ return 0;
+}
+
+static int vb_ioctl_create_pool(void __user *user_cmd)
+{
+ int ret = 0;
+ struct esVB_CREATE_POOL_CMD_S cmd;
+ struct esVB_CREATE_POOL_REQ_S *req;
+ struct esVB_CREATE_POOL_RESP_S *rsp;
+ struct esVB_POOL_CONFIG_S *pool_cfg;
+ struct esVB_K_POOL_INFO_S *pool = NULL;
+
+ if (copy_from_user(&cmd, user_cmd, sizeof(cmd))) {
+ ret = -EFAULT;
+ goto out_free;
+ }
+ req = &cmd.PoolReq;
+ pool_cfg = &req->req;
+ ret = vb_pool_config_check(pool_cfg);
+ if (ret) {
+ goto out_free;
+ }
+ ret = mmz_vb_do_create_pool(pool_cfg, &pool);
+ if (ret) {
+ goto out_free;
+ }
+ pool->enVbUid = VB_UID_PRIVATE;
+ rsp = &cmd.PoolResp;
+ rsp->PoolId = pool->poolId;
+ dev_dbg(mmz_vb_dev, "[%s %d]:create pool, PoolId %d!\n",__func__,__LINE__, rsp->PoolId);
+ if (copy_to_user(user_cmd, &cmd, sizeof(cmd)))
+ ret = -EFAULT;
+ else
+ ret = 0;
+
+out_free:
+ return ret;
+}
+
+/**
+ * mmz_vb_do_destory_pool - do the pool destory operation
+ * @pool: The pool
+ * @is_lock: when set true, will lock the idr when remove the idr id.
+ * @is_force: when set true, will still destory the bool even the bitmap is not empty.
+ */
+static int mmz_vb_do_destory_pool(struct esVB_K_POOL_INFO_S *pool, bool is_lock, bool is_force)
+{
+ struct esVB_POOL_CONFIG_S *poolCfg = &pool->poolCfg;
+ const char *memBlkName = poolCfg->mmzName;
+ struct mem_block *memblock = NULL;
+ struct esVB_K_BLOCK_INFO_S *blocks = NULL;
+ int ret = 0;
+ int i;
+
+ // find the memblock
+ memblock = vb_get_memblock(memBlkName);
+ if (NULL == memblock) {
+ vb_err("%s NOT found!\n", memBlkName);
+ return -EINVAL;
+ }
+
+ if (!bitmap_empty(pool->bitmap, pool->poolCfg.blkCnt)) {
+ if (true == is_force) {
+ dev_info(mmz_vb_dev, "%s %d, non-empty pool, still destory it!\n",__func__,__LINE__);
+ } else {
+ dev_info(mmz_vb_dev, "%s %d, non-empty pool, can not destory!\n",__func__,__LINE__);
+ ret = -ENOTEMPTY;
+ goto out;
+ }
+ }
+
+
+ blocks = pool->blocks;
+ for (i = 0; i < poolCfg->blkCnt; i++) {
+ vb_blk_pages_release(memblock, &blocks[i]);
+ }
+ mmz_vb_remove_pool_id(pool, is_lock);
+ if (pool->enVbUid >= VB_UID_COMMON && pool->enVbUid < VB_UID_MAX) {
+ hash_del(&pool->node);
+ }
+ bitmap_free(pool->bitmap);
+ vfree(pool->blocks);
+out:
+ if (0 == ret) {
+ devm_kfree(mmz_vb_dev, pool);
+ }
+ return ret;
+}
+
+static int vb_ioctl_destory_pool(void __user *user_cmd)
+{
+ int ret = 0;
+ struct esVB_K_POOL_INFO_S *pool = NULL;
+ struct esVB_DESTORY_POOL_CMD_S cmd;
+ struct esVB_DESTORY_POOL_REQ_S *req = NULL;
+ struct mmz_vb_priv *vb_priv = g_mmz_vb_priv;
+ struct esVB_K_MMZ_S *partitions = &vb_priv->partitions;
+
+ if (copy_from_user(&cmd, user_cmd, sizeof(cmd))) {
+ return -EFAULT;
+ }
+ req = &cmd.req;
+ dev_dbg(mmz_vb_dev, "[%s %d]:destory pool, PoolId %d!\n",__func__,__LINE__, req->PoolId);
+ down_write(&partitions->idr_lock);
+ ret = vb_find_pool_by_id_unlock(req->PoolId, &pool);
+ if (ret) {
+ up_write(&partitions->idr_lock);
+ dev_err(mmz_vb_dev, "%s %d, faild to find pool, PoolId %d\n",
+ __func__,__LINE__, req->PoolId);
+ return ret;
+ }
+ ret = mmz_vb_do_destory_pool(pool, false, false);
+ if (-ENOTEMPTY == ret) {
+ set_bit(MMZ_VB_POOL_FLAG_DESTORY, &pool->flag);
+ up_write(&partitions->idr_lock);
+ dev_info(mmz_vb_dev, "%s %d, pool %d not empty, waiting to destory\n",
+ __func__,__LINE__, req->PoolId);
+ return 0;
+ } else if (ret) {
+ up_write(&partitions->idr_lock);
+ dev_err(mmz_vb_dev, "%s %d, faild to destory pool, PoolId %d\n",
+ __func__,__LINE__, req->PoolId);
+ return ret;
+ }
+ up_write(&partitions->idr_lock);
+ return 0;
+}
+
+/*check whether the VbConfig is legal*/
+static int vb_config_check(struct esVB_CONFIG_S *pVbConfig)
+{
+ int i;
+ struct esVB_POOL_CONFIG_S *pool_cfg = NULL;
+ int ret;
+
+ if (pVbConfig->poolCnt > ES_VB_MAX_MOD_POOL) {
+ dev_err(mmz_vb_dev, "%s %d, poolCnt %d exceed the limit %d!\n",
+ __func__,__LINE__, pVbConfig->poolCnt, ES_VB_MAX_MOD_POOL);
+ return -EINVAL;
+ }
+ for (i = 0; i < pVbConfig->poolCnt; i++) {
+ pool_cfg = &pVbConfig->poolCfgs[i];
+ ret = vb_pool_config_check(pool_cfg);
+ if (0 != ret) {
+ return ret;
+ }
+ }
+ return 0;
+}
+
+static int vb_ioctl_set_config(void __user *user_cmd)
+{
+ struct esVB_SET_CFG_CMD_S *cmd;
+ struct esVB_SET_CFG_REQ_S *req;
+ enum esVB_UID_E enVbUid;
+ struct esVB_CONFIG_S *pVbConfig = NULL;
+ struct esVB_CONFIG_S *vb_cfg = NULL;
+ struct mmz_vb_priv *vb_priv = g_mmz_vb_priv;
+ int ret = 0;
+
+ cmd = devm_kzalloc(mmz_vb_dev, sizeof(*cmd), GFP_KERNEL);
+ if (!cmd) {
+ dev_err(mmz_vb_dev, "%s %d, uid %d, failed to alloc memory!\n",
+ __func__,__LINE__, enVbUid);
+ ret = -ENOMEM;
+ goto out;
+ }
+ if (copy_from_user(cmd, user_cmd, sizeof(*cmd))) {
+ ret = -EFAULT;
+ goto out_free_cmd;
+ }
+ req = &cmd->CfgReq;
+ enVbUid = req->uid;
+ pVbConfig = &req->cfg;
+ if (enVbUid <= VB_UID_PRIVATE || enVbUid >= VB_UID_MAX) {
+ dev_err(mmz_vb_dev, "%s %d, invaild uid %d!\n", __func__,__LINE__, enVbUid);
+ ret = -EFAULT;
+ goto out_free_cmd;
+ }
+ ret = vb_config_check(pVbConfig);
+ if (ret) {
+ dev_err(mmz_vb_dev, "%s %d, uid %d, vbConfig check fail!\n",
+ __func__,__LINE__, enVbUid);
+ goto out_free_cmd;
+ }
+ mutex_lock(&vb_priv->cfg_lock[enVbUid]);
+ if (NULL != vb_priv->pVbConfig[enVbUid]) {
+ if (test_bit(MMZ_VB_CFG_FLAG_INIT, &vb_priv->cfg_flag[enVbUid])) {
+ dev_err(mmz_vb_dev, "%s %d, uid %d cfg already exist and init!\n",
+ __func__,__LINE__, enVbUid);
+ ret = -EFAULT;
+ goto out_unlock;
+ } else {
+ /*release the old config*/
+ devm_kfree(mmz_vb_dev, vb_priv->pVbConfig[enVbUid]);
+ vb_priv->pVbConfig[enVbUid] = NULL;
+ }
+ }
+ vb_cfg = devm_kzalloc(mmz_vb_dev, sizeof(struct esVB_CONFIG_S), GFP_KERNEL);
+ if (!vb_cfg) {
+ dev_err(mmz_vb_dev, "%s %d, uid %d, failed to alloc memory!\n",
+ __func__,__LINE__, enVbUid);
+ ret = -ENOMEM;
+ goto out_unlock;
+ }
+ memcpy(vb_cfg, pVbConfig, sizeof(struct esVB_CONFIG_S));
+ vb_priv->pVbConfig[enVbUid] = vb_cfg;
+out_unlock:
+ mutex_unlock(&vb_priv->cfg_lock[enVbUid]);
+out_free_cmd:
+ devm_kfree(mmz_vb_dev, cmd);
+out:
+ return ret;
+}
+
+static int vb_ioctl_get_config(void __user *user_cmd)
+{
+ struct esVB_GET_CFG_CMD_S *cmd;
+ struct esVB_GET_CFG_REQ_S *req;
+ struct esVB_GET_CFG_RSP_S *rsp;
+ enum esVB_UID_E enVbUid;
+ struct esVB_CONFIG_S *vb_cfg = NULL;
+ struct mmz_vb_priv *vb_priv = g_mmz_vb_priv;
+ int ret = 0;
+
+ cmd = devm_kzalloc(mmz_vb_dev, sizeof(*cmd), GFP_KERNEL);
+ if (!cmd) {
+ dev_err(mmz_vb_dev, "%s %d, uid %d, failed to alloc memory!\n",
+ __func__,__LINE__, enVbUid);
+ ret = -ENOMEM;
+ goto out;
+ }
+
+ if (copy_from_user(cmd, user_cmd, sizeof(*cmd))) {
+ ret = -EFAULT;
+ goto out_free_cmd;
+ }
+ req = &cmd->req;
+ enVbUid = req->uid;
+ if (enVbUid <= VB_UID_PRIVATE || enVbUid >= VB_UID_MAX) {
+ dev_err(mmz_vb_dev, "%s %d, invaild uid %d!\n",__func__,__LINE__, enVbUid);
+ ret = -EFAULT;
+ goto out_free_cmd;
+ }
+ mutex_lock(&vb_priv->cfg_lock[enVbUid]);
+ vb_cfg = vb_priv->pVbConfig[enVbUid];
+ if (NULL == vb_cfg) {
+ dev_err(mmz_vb_dev, "%s %d, uid %d cfg not exist!\n", __func__,__LINE__, enVbUid);
+ ret = -EFAULT;
+ goto out_unlock;
+ }
+ rsp = &cmd->rsp;
+ memcpy(&rsp->cfg, vb_cfg, sizeof(struct esVB_CONFIG_S));
+ if (copy_to_user(user_cmd, cmd, sizeof(*cmd))) {
+ ret = -EFAULT;
+ goto out_unlock;
+ }
+out_unlock:
+ mutex_unlock(&vb_priv->cfg_lock[enVbUid]);
+out_free_cmd:
+ devm_kfree(mmz_vb_dev, cmd);
+out:
+ return ret;
+}
+
+static int vb_ioctl_init_config(void __user *user_cmd)
+{
+ VB_INIT_CFG_CMD_S cmd;
+ VB_INIT_CFG_REQ_S *req = NULL;
+ enum esVB_UID_E enVbUid;
+ struct esVB_CONFIG_S *vb_cfg = NULL;
+ struct mmz_vb_priv *vb_priv = g_mmz_vb_priv;
+ int i;
+ int ret = 0;
+ struct esVB_POOL_CONFIG_S *pool_cfg;
+ struct esVB_K_POOL_INFO_S *pool[ES_VB_MAX_MOD_POOL] = {NULL};
+
+ if (copy_from_user(&cmd, user_cmd, sizeof(cmd))) {
+ ret = -EFAULT;
+ goto out;
+ }
+ req = &cmd.req;
+ enVbUid = req->uid;
+
+ if (enVbUid < VB_UID_COMMON || enVbUid >= VB_UID_MAX) {
+ dev_err(mmz_vb_dev, "%s %d, invaild uid %d!\n",__func__,__LINE__, enVbUid);
+ ret = -EFAULT;
+ goto out;
+ }
+ mutex_lock(&vb_priv->cfg_lock[enVbUid]);
+ vb_cfg = vb_priv->pVbConfig[enVbUid];
+ if (NULL == vb_cfg) {
+ dev_err(mmz_vb_dev, "%s %d, uid %d cfg not exist!\n", __func__,__LINE__, enVbUid);
+ ret = -EFAULT;
+ goto out_unlock;
+ }
+ if (test_bit(MMZ_VB_CFG_FLAG_INIT, &vb_priv->cfg_flag[enVbUid])) {
+ dev_err(mmz_vb_dev, "%s %d, uid %d cfg already initialized!\n", __func__,__LINE__, enVbUid);
+ ret = -EPERM;
+ goto out_unlock;
+ }
+
+ for (i = 0; i < vb_cfg->poolCnt; i++) {
+ pool_cfg = &vb_cfg->poolCfgs[i];
+ ret = mmz_vb_do_create_pool(pool_cfg, &pool[i]);
+ if (0 != ret) {
+ while(--i >= 0) {
+ ret = mmz_vb_do_destory_pool(pool[i], true, false);
+ if (ret) {
+ dev_err(mmz_vb_dev, "%s %d, faild to destory pool!\n",
+ __func__,__LINE__);
+ }
+ }
+ dev_err(mmz_vb_dev, "%s %d, faild to create pool!\n",__func__, __LINE__);
+ goto out_unlock;
+ }
+ mmz_pool_insert_list(pool[i], enVbUid);
+ pool[i]->enVbUid = enVbUid;
+ }
+ set_bit(MMZ_VB_CFG_FLAG_INIT, &vb_priv->cfg_flag[enVbUid]);
+out_unlock:
+ mutex_unlock(&vb_priv->cfg_lock[enVbUid]);
+out:
+ return ret;
+}
+
+static int vb_ioctl_uninit_config(void __user *user_cmd)
+{
+ struct esVB_UNINIT_CFG_CMD_S cmd;
+ enum esVB_UID_E enVbUid;
+ int ret;
+ struct mmz_vb_priv *vb_priv = g_mmz_vb_priv;
+ struct esVB_K_POOL_INFO_S *pool = NULL;
+ unsigned long bkt = 0;
+ struct hlist_node *tmp_node = NULL;
+
+ if (copy_from_user(&cmd, user_cmd, sizeof(cmd))) {
+ return -EFAULT;
+ }
+ enVbUid = cmd.req.uid;
+
+ if (enVbUid <= VB_UID_PRIVATE || enVbUid >= VB_UID_MAX) {
+ dev_err(mmz_vb_dev, "%s %d, invaild uid %d!\n",__func__,__LINE__, enVbUid);
+ return -EFAULT;
+ }
+ mutex_lock(&vb_priv->cfg_lock[enVbUid]);
+ if (!test_bit(MMZ_VB_CFG_FLAG_INIT, &vb_priv->cfg_flag[enVbUid])) {
+ dev_err(mmz_vb_dev, "%s %d, uid %d cfg not initialized!\n", __func__,__LINE__, enVbUid);
+ mutex_unlock(&vb_priv->cfg_lock[enVbUid]);
+ return -EINVAL;
+ }
+ mutex_unlock(&vb_priv->cfg_lock[enVbUid]);
+
+ down_write(&vb_priv->pool_lock[enVbUid]);
+ hash_for_each_safe(vb_priv->ht[enVbUid], bkt, tmp_node, pool, node) {
+ ret = mmz_vb_do_destory_pool(pool, true, false);
+ if (ret) {
+ dev_err(mmz_vb_dev, "%s %d, faild to destory pool, PoolId %d, enVbUid %d\n",
+ __func__,__LINE__, pool->poolId, enVbUid);
+ up_write(&vb_priv->pool_lock[enVbUid]);
+ return ret;
+ }
+ }
+ up_write(&vb_priv->pool_lock[enVbUid]);
+
+ mutex_lock(&vb_priv->cfg_lock[enVbUid]);
+ devm_kfree(mmz_vb_dev, vb_priv->pVbConfig[enVbUid]);
+ vb_priv->pVbConfig[enVbUid] = NULL;
+ clear_bit(MMZ_VB_CFG_FLAG_INIT, &vb_priv->cfg_flag[enVbUid]);
+ mutex_unlock(&vb_priv->cfg_lock[enVbUid]);
+ return 0;
+}
+
+static long mmz_vb_unlocked_ioctl(struct file *file, unsigned int cmd, unsigned long arg)
+{
+ int ret = 0;
+ void __user *argp;
+
+ argp = (void __user *)arg;
+ switch (cmd) {
+ case MMZ_VB_IOCTL_GET_BLOCK:
+ return vb_ioctl_get_blk(argp);
+ case MMZ_VB_IOCTL_CREATE_POOL:
+ return vb_ioctl_create_pool(argp);
+ case MMZ_VB_IOCTL_DESTORY_POOL:
+ return vb_ioctl_destory_pool(argp);
+ case MMZ_VB_IOCTL_SET_CFG:
+ return vb_ioctl_set_config(argp);
+ case MMZ_VB_IOCTL_GET_CFG:
+ return vb_ioctl_get_config(argp);
+ case MMZ_VB_IOCTL_INIT_CFG:
+ return vb_ioctl_init_config(argp);
+ case MMZ_VB_IOCTL_UNINIT_CFG:
+ return vb_ioctl_uninit_config(argp);
+ case MMZ_VB_IOCTL_POOL_SIZE:
+ return vb_ioctl_pool_size(argp);
+#if 0
+ case MMZ_VB_IOCTL_FLUSH_POOL:
+ return vb_ioctl_flush_pool(argp);
+#endif
+ case MMZ_VB_IOCTL_BLOCK_TO_POOL:
+ return vb_ioctl_blk_to_pool(argp);
+ case MMZ_VB_IOCTL_GET_BLOCK_OFFSET:
+ return vb_ioctl_get_blk_offset(argp);
+ case MMZ_VB_IOCTL_SPLIT_DMABUF:
+ return vb_ioctl_split_dmabuf(argp);
+ case MMZ_VB_IOCTL_DMABUF_REFCOUNT:
+ return vb_ioctl_get_dmabuf_refcnt(argp);
+ case MMZ_VB_IOCTL_RETRIEVE_MEM_NODE:
+ return vb_ioctl_retrieve_mem_node(argp);
+ case MMZ_VB_IOCTL_DMABUF_SIZE:
+ return vb_ioctl_get_dmabuf_size(argp);
+ default:
+ pr_debug("Invalid IOCTL CMD!!!\n");
+ return -EINVAL;
+ }
+ pr_debug("%s:%d, success!\n", __func__, __LINE__);
+ return ret;
+}
+
+static int mmz_vb_open(struct inode *inode, struct file *file)
+{
+
+ pr_debug("%s:%d, success!\n", __func__, __LINE__);
+
+ return 0;
+}
+
+static int mmz_vb_release(struct inode *inode, struct file *file)
+{
+ pr_debug("%s:%d, success!\n", __func__, __LINE__);
+
+ return 0;
+}
+
+/* mem = mmap(0, len, PROT_READ | PROT_WRITE, MAP_SHARED, fd, addr_pha);
+* vma->vm_pgoff indicats the pool ID
+*/
+static int mmz_vb_mmap_pool(struct file *file, struct vm_area_struct *vma)
+{
+ int ret = 0;
+ size_t size = vma->vm_end - vma->vm_start;
+ VB_POOL poolId = (VB_POOL)vma->vm_pgoff;
+ unsigned long addr = vma->vm_start;
+ struct esVB_K_POOL_INFO_S *pPool = NULL;
+ struct esVB_K_BLOCK_INFO_S *pBlk = NULL;
+ u64 poolSize, blkSize;
+ u32 i;
+
+ ret = vb_find_pool_by_id(poolId, &pPool);
+ if (ret)
+ return ret;
+
+ poolSize = do_vb_pool_size(pPool);
+ /* is the mmap size equal to poolSize? */
+ if (size != poolSize)
+ return -EINVAL;
+
+ /* pool is mmapped as uncached memory */
+ #ifndef QEMU_DEBUG
+ vma->vm_page_prot = pgprot_dmacoherent(vma->vm_page_prot);
+ #endif
+
+ blkSize = pPool->poolCfg.blkSize;
+ pBlk = pPool->blocks;
+ for (i = 0; i < pPool->poolCfg.blkCnt; i++) {
+ struct sg_table *table = &pBlk->sg_table;
+
+ /* mmap for one block */
+ struct scatterlist *sg;
+ int j;
+
+ for_each_sg(table->sgl, sg, table->orig_nents, j) {
+ struct page *page = sg_page(sg);
+ ret = remap_pfn_range(vma, addr, page_to_pfn(page), sg->length,
+ vma->vm_page_prot);
+ if (ret)
+ return ret;
+ addr += sg->length;
+ if (addr >= vma->vm_end)
+ return 0;
+ }
+ pBlk++;
+ }
+
+ pr_debug("%s:%d, success!\n", __func__, __LINE__);
+
+ return ret;
+}
+
+static struct file_operations mmz_vb_fops = {
+ .owner = THIS_MODULE,
+ .llseek = no_llseek,
+ .unlocked_ioctl = mmz_vb_unlocked_ioctl,
+ .open = mmz_vb_open,
+ .release = mmz_vb_release,
+ .mmap = mmz_vb_mmap_pool,
+};
+
+static struct miscdevice mmz_vb_miscdev = {
+ .minor = MISC_DYNAMIC_MINOR,
+ .name = DRIVER_NAME,
+ .fops = &mmz_vb_fops,
+};
+
+static char es_mmz_name_prefix[] = "mmz_nid_";
+static int mmz_vb_init_partitions(void)
+{
+ int ret = 0;
+ struct mmz_vb_priv *mmz_vb_priv = g_mmz_vb_priv;
+ struct esVB_K_MMZ_S *partitions;
+
+ if (NULL == mmz_vb_priv)
+ return -EFAULT;
+
+ partitions = &mmz_vb_priv->partitions;;
+ init_rwsem(&partitions->idr_lock);
+ idr_init(&partitions->pool_idr);
+
+ partitions->partCnt = mmz_vb_init_memory_region();
+ if (partitions->partCnt == 0) {
+ vb_err("No VB memory block was found or correctly initialized!\n");
+ ret = -EFAULT;
+ }
+
+ return ret;
+}
+
+static int mmz_vb_idr_iterate_show(int id, void *p, void *data)
+{
+ struct esVB_K_POOL_INFO_S *pool = (struct esVB_K_POOL_INFO_S *)p;
+ struct esVB_POOL_CONFIG_S *pool_cfg;
+ es_proc_entry_t *s = (es_proc_entry_t *)data;
+
+ spin_lock(&pool->lock);
+ pool_cfg = &pool->poolCfg;
+ es_seq_printf(s, "\t Uid %d, PoolId %d, blkSize 0x%llx, blkCnt %d, "
+ "RemapMode %d, mmzName %s, allocated blkCnt %d\n\r", pool->enVbUid,
+ pool->poolId, pool_cfg->blkSize, pool_cfg->blkCnt,
+ pool_cfg->enRemapMode, pool_cfg->mmzName,
+ pool_cfg->blkCnt - vb_pool_get_free_block_cnt_unlock(pool));
+ spin_unlock(&pool->lock);
+ return 0;
+}
+
+static int mmz_vb_proc_show(es_proc_entry_t *s)
+{
+ int i;
+ struct mmz_vb_priv *vb_priv = g_mmz_vb_priv;
+ struct esVB_K_MMZ_S *partitions = &vb_priv->partitions;
+ unsigned long numFreePages = 0;
+ struct mem_block *memblock = NULL, *rsvmem_block = NULL;
+ int ret;
+
+ es_seq_printf(s, "\nModule: [VB], Build Time[xx]\n");
+ /*
+ use es_seq_printf to show more debug info
+ */
+ es_seq_printf(s, "-----MMZ REGION CONFIG-----\n\r");
+ for (i = 0; i < partitions->partCnt; i++) {
+ rsvmem_block = partitions->mem_blocks[i];
+ memblock = vb_get_memblock(rsvmem_block->name);
+ if (NULL == memblock) {
+ vb_err("%s NOT found!\n", rsvmem_block->name);
+ return -EINVAL;
+ }
+ numFreePages = es_num_free_pages(memblock);
+ es_seq_printf(s, "\tmemblock: %s, total size(0x%lx), free mem size(0x%lx)\n\r",
+ rsvmem_block->name,
+ memblock->page_num << PAGE_SHIFT,
+ numFreePages << PAGE_SHIFT);
+ }
+ es_seq_printf(s, "-----POOL CONFIG-----\n\r");
+ ret = idr_for_each(&partitions->pool_idr, mmz_vb_idr_iterate_show, s);
+ if (ret) {
+ dev_err(mmz_vb_dev, "%s %d, failed to iterate vb pool ret %d\n",
+ __func__,__LINE__, ret);
+ return ret;
+ }
+ return 0;
+}
+
+int mmz_vb_proc_store(struct es_proc_dir_entry *entry, const char *buf,
+ int count, long long *ppos)
+{
+ int ret;
+
+ ret = mmz_vb_pool_exit();
+ if (0 != ret) {
+ dev_err(mmz_vb_dev, "%s %d, failed to release vb pool "
+ "when exit, ret %d\n", __func__,__LINE__, ret);
+ }
+ return count;
+}
+
+void mmz_vb_vb_dbg_init(void)
+{
+ es_proc_entry_t *proc = NULL;
+
+ proc = es_create_proc_entry(PROC_ENTRY_VB, NULL);
+
+ if (proc == NULL) {
+ vb_err("Kernel: Register vb proc failed!\n");
+ return;
+ }
+ proc->read = mmz_vb_proc_show;
+ /*NULL means use the default routine*/
+ proc->write = mmz_vb_proc_store;
+ proc->open = NULL;
+}
+
+static int __init mmz_vb_init(void)
+{
+ int i;
+ int ret = 0;
+ struct device *dev;
+
+ g_mmz_vb_priv = kzalloc(sizeof(struct mmz_vb_priv), GFP_KERNEL);
+ if (!g_mmz_vb_priv) {
+ vb_err("Failed to alloc priv data for mmz_vb driver!!!\n");
+ return -ENOMEM;
+ }
+ ret = misc_register(&mmz_vb_miscdev);
+ if(ret) {
+ vb_err ("cannot register miscdev (err=%d)\n", ret);
+ goto free_vb_priv;
+ }
+ mmz_vb_dev = mmz_vb_miscdev.this_device;
+ g_mmz_vb_priv->dev = mmz_vb_dev;
+ for (i = 0; i < VB_UID_MAX; i++) {
+ hash_init(g_mmz_vb_priv->ht[i]);
+ mutex_init(&g_mmz_vb_priv->cfg_lock[i]);
+ init_rwsem(&g_mmz_vb_priv->pool_lock[i]);
+ }
+
+ ret = mmz_vb_init_partitions();
+ if (ret) {
+ goto deregister_vb;
+ }
+ atomic_set(&g_mmz_vb_priv->allocBlkcnt, 0);
+
+ mmz_vb_vb_dbg_init();
+
+ dev = mmz_vb_dev;
+ if (!dev->dma_mask) {
+ dev->dma_mask = &dev->coherent_dma_mask;
+ }
+ ret = dma_set_mask_and_coherent(dev, DMA_BIT_MASK(48));
+ if (ret)
+ vb_err("Unable to set coherent mask\n");
+ return 0;
+
+deregister_vb:
+ misc_deregister(&mmz_vb_miscdev);
+free_vb_priv:
+ kfree(g_mmz_vb_priv);
+ return ret;
+}
+
+static int mmz_vb_pool_exit(void)
+{
+ struct mmz_vb_priv *vb_priv = g_mmz_vb_priv;
+ struct esVB_K_MMZ_S *partitions = &vb_priv->partitions;
+ struct esVB_K_POOL_INFO_S *pool = NULL;
+ int ret = 0;
+ u32 id = 0;
+ int i;
+
+ down_write(&partitions->idr_lock);
+ idr_for_each_entry(&partitions->pool_idr, pool, id) {
+ ret = mmz_vb_do_destory_pool(pool, false, true);
+ if (ret) {
+ dev_err(mmz_vb_dev, "%s %d, failed to destory vb pool, ret %d\n",
+ __func__,__LINE__, ret);
+ continue;
+ }
+ }
+
+ up_write(&partitions->idr_lock);
+
+ atomic_set(&vb_priv->allocBlkcnt, 0);
+ for (i = 0; i < VB_UID_MAX; i++) {
+ if (NULL != vb_priv->pVbConfig[i]) {
+ devm_kfree(mmz_vb_dev, vb_priv->pVbConfig[i]);
+ vb_priv->pVbConfig[i] = NULL;
+ }
+ }
+ memset(vb_priv->cfg_flag, 0, sizeof(unsigned long) * VB_UID_MAX);
+ return 0;
+}
+
+static void __exit mmz_vb_exit(void)
+{
+ struct mmz_vb_priv *vb_priv = g_mmz_vb_priv;
+ int ret = 0;
+
+ ret = mmz_vb_pool_exit();
+ if (0 != ret) {
+ dev_err(mmz_vb_dev, "%s %d, failed to release vb pool "
+ "when exit, ret %d\n", __func__,__LINE__, ret);
+ }
+ es_remove_proc_entry(PROC_ENTRY_VB, NULL);
+ misc_deregister(&mmz_vb_miscdev);
+ kfree(vb_priv);
+}
+
+module_init(mmz_vb_init);
+module_exit(mmz_vb_exit);
+
+MODULE_DESCRIPTION("MMZ VB Driver");
+MODULE_AUTHOR("Lin MIn <linmin@eswincomputing.com>");
+MODULE_LICENSE("GPL v2");
+
+
+static int vb_find_pool_by_id_unlock(VB_POOL poolId, struct esVB_K_POOL_INFO_S **ppPool)
+{
+ struct mmz_vb_priv *vb_priv = g_mmz_vb_priv;
+ struct esVB_K_MMZ_S *partitions = &vb_priv->partitions;
+ struct esVB_K_POOL_INFO_S *pool = NULL;
+
+ pool = idr_find(&partitions->pool_idr, poolId);
+ if (!pool) {
+ dev_err(mmz_vb_dev, "%s %d, faild to find pool by id %d\n",
+ __func__,__LINE__, poolId);
+ return -EINVAL;
+ }
+ *ppPool = pool;
+ return 0;
+}
+
+static int vb_find_pool_by_id(VB_POOL poolId, struct esVB_K_POOL_INFO_S **ppPool)
+{
+ struct mmz_vb_priv *vb_priv = g_mmz_vb_priv;
+ struct esVB_K_MMZ_S *partitions = &vb_priv->partitions;
+ int ret;
+
+ down_read(&partitions->idr_lock);
+ ret = vb_find_pool_by_id_unlock(poolId, ppPool);
+ up_read(&partitions->idr_lock);
+ return ret;
+}
+
+static int vb_pool_size(VB_POOL poolId, u64 *pPoolSize)
+{
+ int ret = 0;
+ struct esVB_K_POOL_INFO_S *pPool;
+
+ ret = vb_find_pool_by_id(poolId, &pPool);
+ if (ret) {
+ vb_info("failed to find pool %d\n", poolId);
+ return ret;
+ }
+
+ *pPoolSize = do_vb_pool_size(pPool);
+
+ return ret;
+}
+#if 0
+static int vb_flush_pool(struct esVB_FLUSH_POOL_CMD_S *flushPoolCmd)
+{
+ int ret = 0;
+ struct esVB_K_POOL_INFO_S *pPool = NULL;
+ struct esVB_K_BLOCK_INFO_S *pBlk = NULL;
+ u64 blkSize, poolSize = 0;
+ u64 offset_inPool = 0, offset_inBlk = 0, size, left_size = 0;
+ u64 phy_addr;
+ u32 i;
+
+ ret = vb_find_pool_by_id(flushPoolCmd->poolId, &pPool);
+ if (ret) {
+ vb_info("%s,failed to find pool %d\n", __func__, flushPoolCmd->poolId);
+ return ret;
+ }
+
+ poolSize = do_vb_pool_size(pPool);
+ if ((flushPoolCmd->offset + flushPoolCmd->size - 1) >= poolSize)
+ return -EINVAL;
+
+ // find the block according to the offset
+ blkSize = pPool->poolCfg.blkSize;
+ pBlk = pPool->blocks;
+ left_size = flushPoolCmd->size;
+ for (i = 0; i < pPool->poolCfg.blkCnt; i++) {
+ if ((offset_inPool + blkSize -1) >= flushPoolCmd->offset)
+ break;
+ offset_inPool += blkSize;
+ pBlk++;
+ }
+ offset_inBlk = flushPoolCmd->offset - offset_inPool;
+ for (; i < pPool->poolCfg.blkCnt; i++) {
+ struct page *page = pBlk->cma_pages;
+ size = min(left_size, (blkSize - offset_inBlk));
+ phy_addr = page_to_phys(page) + offset_inBlk;
+ arch_sync_dma_for_device(phy_addr, size, DMA_TO_DEVICE);
+ left_size -= size;
+ if (left_size == 0)
+ break;
+ pBlk++;
+ offset_inBlk = 0;
+ }
+
+ return ret;
+}
+#endif
+static int vb_pool_get_free_block_cnt_unlock(struct esVB_K_POOL_INFO_S *pool)
+{
+ int count = 0;
+ int start = 0;
+ struct esVB_POOL_CONFIG_S *pool_cfg = &pool->poolCfg;
+ int nr;
+
+ while (true) {
+ nr = find_next_zero_bit(pool->bitmap, pool_cfg->blkCnt, start);
+ if (likely(nr < pool_cfg->blkCnt)) {
+ count++;
+ start = nr + 1;
+ } else {
+ break;
+ }
+ }
+ return count;
+}
+
+static int vb_pool_get_block(struct esVB_K_POOL_INFO_S *pool,
+ struct esVB_K_BLOCK_INFO_S **ppBlk)
+{
+ unsigned int nr = -1U;
+ struct esVB_POOL_CONFIG_S *pool_cfg;
+ int ret = -EINVAL;
+ struct mmz_vb_priv *vb_priv = g_mmz_vb_priv;
+
+ spin_lock(&pool->lock);
+ pool_cfg = &pool->poolCfg;
+ nr = find_next_zero_bit(pool->bitmap, pool_cfg->blkCnt, 0);
+ if (likely(nr < pool_cfg->blkCnt)) {
+ ret = 0;
+ *ppBlk = &pool->blocks[nr];
+ bitmap_set(pool->bitmap, nr, 1);
+ if (atomic_inc_return(&vb_priv->allocBlkcnt) == 1) {
+ __module_get(THIS_MODULE);
+ }
+ } else {
+ dev_warn(mmz_vb_dev, "%s %d, pool %d used up, blkSize 0x%llx,"
+ "blkCnt 0x%x\n",__func__,__LINE__, pool->poolId,
+ pool_cfg->blkSize, pool_cfg->blkCnt);
+ }
+ spin_unlock(&pool->lock);
+ return ret;
+}
+
+static int vb_get_block(struct esVB_GET_BLOCK_CMD_S *getBlkCmd,
+ struct esVB_K_BLOCK_INFO_S **ppBlk)
+{
+ int ret = -EINVAL;
+ struct mmz_vb_priv *vb_priv = g_mmz_vb_priv;
+ struct esVB_K_MMZ_S *partitions = &vb_priv->partitions;
+ struct esVB_GET_BLOCK_REQ_S *req = &getBlkCmd->getBlkReq;
+ struct esVB_K_POOL_INFO_S *pool = NULL, *pool_tmp = NULL;
+ struct esVB_POOL_CONFIG_S *pool_cfg;
+ unsigned long bkt = 0;
+
+ if (VB_UID_PRIVATE == req->uid) {
+ down_read(&partitions->idr_lock);
+ ret = vb_find_pool_by_id_unlock(req->poolId, &pool);
+ if (ret) {
+ up_read(&partitions->idr_lock);
+ dev_err(mmz_vb_dev, "%s %d, failed to find pool by id %d!\n",__func__,__LINE__, req->poolId);
+ return -EINVAL;
+ }
+ if (test_bit(MMZ_VB_POOL_FLAG_DESTORY, &pool->flag)) {
+ up_read(&partitions->idr_lock);
+ dev_err(mmz_vb_dev, "%s %d, pool %d is in destory state, not allow "
+ "to alloc block!\n",__func__,__LINE__, req->poolId);
+ return -ENOTSUPP;
+ }
+ pool_cfg = &pool->poolCfg;
+ if (req->blkSize > pool_cfg->blkSize) {
+ up_read(&partitions->idr_lock);
+ dev_err(mmz_vb_dev, "%s %d, pool blkSize 0x%llx is "
+ "smaller than request size 0x%llx\n",__func__,__LINE__,
+ pool_cfg->blkSize, req->blkSize);
+ return -EINVAL;
+ }
+ ret = vb_pool_get_block(pool, ppBlk);
+ up_read(&partitions->idr_lock);
+ } else if (req->uid >= VB_UID_COMMON && req->uid < VB_UID_MAX) {
+ down_read(&vb_priv->pool_lock[req->uid]);
+ /*try to get block for the exact block size */
+ hash_for_each_possible(vb_priv->ht[req->uid], pool, node, PAGE_ALIGN(req->blkSize)) {
+ pool_cfg = &pool->poolCfg;
+ if (PAGE_ALIGN(req->blkSize) == pool_cfg->blkSize &&
+ !strcmp(req->mmzName, pool_cfg->mmzName)) {
+ ret = vb_pool_get_block(pool, ppBlk);
+ if (0 == ret) {
+ break;
+ }
+ }
+ }
+ /*try to get block from the pool whose block size > req->blkSize*/
+ if (0 != ret) {
+ hash_for_each(vb_priv->ht[req->uid], bkt, pool, node) {
+ pool_cfg = &pool->poolCfg;
+ if (req->blkSize < pool_cfg->blkSize &&
+ !strcmp(req->mmzName, pool_cfg->mmzName)) {
+ /*get the pool size which is closest to the req->blkSize*/
+ if ((NULL == pool_tmp || (pool_tmp->poolCfg.blkSize > pool->poolCfg.blkSize))
+ && vb_pool_get_free_block_cnt_unlock(pool)) {
+ pool_tmp = pool;
+ }
+ }
+ }
+ if (NULL != pool_tmp) {
+ ret = vb_pool_get_block(pool_tmp, ppBlk);
+ }
+ }
+ up_read(&vb_priv->pool_lock[req->uid]);
+ } else {
+ dev_err(mmz_vb_dev, "%s %d, invaild uid %d\n",__func__,__LINE__, req->uid);
+ }
+ return ret;
+}
+
+static void vb_release_block(struct esVB_K_BLOCK_INFO_S *pBlk)
+{
+ struct esVB_K_POOL_INFO_S *pool;
+ struct mmz_vb_priv *vb_priv = g_mmz_vb_priv;
+ struct esVB_K_MMZ_S *partitions = &vb_priv->partitions;
+ bool need_destory = false;
+ struct rw_semaphore *lock;
+ int ret;
+
+ pool = pBlk->pool;
+
+ lock = VB_UID_PRIVATE == pool->enVbUid ? \
+ &partitions->idr_lock : &vb_priv->pool_lock[pool->enVbUid];
+ /*
+ usually we don't need to destory pool here.
+ so just get read lock first.
+ */
+ down_read(lock);
+ spin_lock(&pool->lock);
+ bitmap_clear(pool->bitmap, pBlk->nr, 1);
+ if (bitmap_empty(pool->bitmap, pool->poolCfg.blkCnt) &&
+ test_bit(MMZ_VB_POOL_FLAG_DESTORY, &pool->flag)) {
+ need_destory = true;
+ }
+ spin_unlock(&pool->lock);
+ up_read(lock);
+ if (atomic_dec_return(&vb_priv->allocBlkcnt) == 0) {
+ module_put(THIS_MODULE);
+ }
+
+ if (true == need_destory) {
+ down_write(lock);
+ ret = mmz_vb_do_destory_pool(pool, false, false);
+ if (ret) {
+ dev_err(mmz_vb_dev, "%s %d, faild to destory pool, enVbUid %d, PoolId %d, ret %d\n",
+ __func__,__LINE__, pool->enVbUid, pool->poolId, ret);
+ }
+ up_write(lock);
+ }
+}
+
+static int vb_is_splitted_blk(int fd, bool *isSplittedBlk)
+{
+ int ret = 0;
+ struct dma_buf *dmabuf;
+
+ /* get dmabuf handle */
+ dmabuf = dma_buf_get(fd);
+ if (IS_ERR(dmabuf)) {
+ return -EINVAL;
+ }
+
+ if (strncmp(dmabuf->exp_name, MMZ_VB_DMABUF_NAME, sizeof(MMZ_VB_DMABUF_NAME))) {
+ vb_err("It's NOT a mmz_vb buffer!!!\n");
+ dma_buf_put(dmabuf);
+ return -EINVAL;
+ }
+
+ if (!strncmp(dmabuf->exp_name, MMZ_VB_DMABUF_SPLITTED_NAME, sizeof(MMZ_VB_DMABUF_SPLITTED_NAME)))
+ *isSplittedBlk = true;
+ else
+ *isSplittedBlk = false;
+
+ dma_buf_put(dmabuf);
+ return ret;
+}
+
+static int vb_blk_to_pool(struct esVB_BLOCK_TO_POOL_CMD_S *blkToPoolCmd)
+{
+ int ret = 0;
+ struct dma_buf *dmabuf;
+ struct mmz_vb_buffer *buffer;
+ struct esw_export_buffer_info *splittedBuffer;
+ struct dma_buf *blkDmabuf;
+ bool isSplittedBlk;
+
+ /* get dmabuf handle */
+ dmabuf = dma_buf_get(blkToPoolCmd->fd);
+ if (IS_ERR(dmabuf)) {
+ return -EINVAL;
+ }
+
+ ret = vb_is_splitted_blk(blkToPoolCmd->fd, &isSplittedBlk);
+ if (ret) {
+ dma_buf_put(dmabuf);
+ ret = -EINVAL;
+ goto out_put_dmabuf;
+ }
+
+ if (true == isSplittedBlk) { // This is a splitted block
+ splittedBuffer = dmabuf->priv;
+ blkDmabuf = dma_buf_get(splittedBuffer->dbuf_fd);
+ if (IS_ERR(blkDmabuf)) {
+ ret = -EINVAL;
+ goto out_put_dmabuf;
+ }
+ buffer = blkDmabuf->priv;
+ blkToPoolCmd->poolId = buffer->pBlk->pool->poolId;
+ dma_buf_put(blkDmabuf);
+ }
+ else { // This is a real block
+ buffer = dmabuf->priv;
+ blkToPoolCmd->poolId = buffer->pBlk->pool->poolId;
+ }
+
+out_put_dmabuf:
+ dma_buf_put(dmabuf);
+ return ret;
+}
+
+static int vb_get_blk_offset(struct esVB_GET_BLOCKOFFSET_CMD_S *getBlkOffsetCmd)
+{
+ int ret = 0;
+ struct dma_buf *dmabuf;
+ struct mmz_vb_buffer *buffer;
+ struct esw_export_buffer_info *splittedBuffer;
+ struct dma_buf *blkDmabuf;
+ __u64 blkSize, offsetInPool;
+
+ bool isSplittedBlk;
+
+ dmabuf = dma_buf_get(getBlkOffsetCmd->fd);
+ if (IS_ERR(dmabuf)) {
+ return -EINVAL;
+ }
+
+ ret = vb_is_splitted_blk(getBlkOffsetCmd->fd, &isSplittedBlk);
+ if (ret) {
+ ret = -EINVAL;
+ goto out_put_dmabuf;
+ }
+
+ if (true == isSplittedBlk) { // It's a splitted block
+ splittedBuffer = dmabuf->priv;
+ blkDmabuf = dma_buf_get(splittedBuffer->dbuf_fd);
+ if (IS_ERR(blkDmabuf)) {
+ ret = -EINVAL;
+ goto out_put_dmabuf;
+ }
+ buffer = blkDmabuf->priv;
+ blkSize = buffer->len;
+ offsetInPool = blkSize * buffer->pBlk->nr + splittedBuffer->slice.offset;
+ dma_buf_put(blkDmabuf);
+ }
+ else { // It's a real block
+ buffer = dmabuf->priv;
+ blkSize = buffer->len;
+ offsetInPool = blkSize * buffer->pBlk->nr;
+ }
+ getBlkOffsetCmd->offset = offsetInPool;
+
+out_put_dmabuf:
+ dma_buf_put(dmabuf);
+
+ return ret;
+}
+
+static int vb_split_dmabuf(struct esVB_SPLIT_DMABUF_CMD_S *splitDmabufCmd)
+{
+ int ret = 0;
+ struct dma_buf *dmabuf;
+ char splittedBuffer_ExpName[ES_MAX_MMZ_NAME_LEN];
+ int i;
+
+ if (!splitDmabufCmd->len)
+ return -EINVAL;
+
+ /* get dmabuf handle */
+ dmabuf = dma_buf_get(splitDmabufCmd->fd);
+ if (IS_ERR(dmabuf)) {
+ return -EINVAL;
+ }
+
+ if (strstr(dmabuf->exp_name, "_splitted")) { // It's a splitted dmabuf already, can't be splitted further
+ vb_err("Can't split a splitted buffer!!!\n");
+ ret = -EINVAL;
+ goto out_put_dmabuf;
+ }
+
+ /* offset and len must be paged aligned */
+ if (!PAGE_ALIGNED(splitDmabufCmd->offset) || !PAGE_ALIGNED(splitDmabufCmd->len)) {
+ vb_err("splitted offset or len is not page aligned!!!\n");
+ ret = -EINVAL;
+ goto out_put_dmabuf;
+ }
+
+ if (splitDmabufCmd->offset + splitDmabufCmd->len > dmabuf->size) {
+ vb_err("Splitted offset(0x%llx)+len(0x%llx) exceed the size(0x%llx) of the original buffer!!!\n",
+ splitDmabufCmd->offset, splitDmabufCmd->len, (__u64)dmabuf->size);
+ ret = -EINVAL;
+ goto out_put_dmabuf;
+ }
+
+ /* Apend "_splitted" to the splitted buffer expname, so that it is identified by the suffix */
+ i = snprintf(splittedBuffer_ExpName, sizeof(splittedBuffer_ExpName), "%s_splitted", dmabuf->exp_name);
+ if (i > sizeof(splittedBuffer_ExpName)) {
+ vb_err("Length of name(%d) for the the splitted buffer exceed the max name len(%ld)!!!\n",
+ i, sizeof(splittedBuffer_ExpName));
+ ret = -EINVAL;
+ goto out_put_dmabuf;
+ }
+
+ splitDmabufCmd->slice_fd = esw_common_dmabuf_split_export(splitDmabufCmd->fd, splitDmabufCmd->offset,
+ splitDmabufCmd->len, dmabuf->file->f_flags, splittedBuffer_ExpName);
+ if (splitDmabufCmd->slice_fd < 0) {
+ vb_err("Failed to split buffer, errVal %d\n", splitDmabufCmd->slice_fd);
+ ret = -EFAULT;
+ goto out_put_dmabuf;
+ }
+
+out_put_dmabuf:
+ dma_buf_put(dmabuf);
+
+ return ret;
+}
+
+static int vb_get_dmabuf_refcnt(struct esVB_DMABUF_REFCOUNT_CMD_S *getDmabufRefCntCmd)
+{
+ int ret = 0;
+ struct dma_buf *dmabuf;
+
+ /* get dmabuf handle */
+ dmabuf = dma_buf_get(getDmabufRefCntCmd->fd);
+ if (IS_ERR(dmabuf)) {
+ return -EINVAL;
+ }
+
+ /* minus 1 because it was +1 by dma_buf_get */
+ getDmabufRefCntCmd->refCnt = file_count(dmabuf->file) - 1;
+
+ dma_buf_put(dmabuf);
+ return ret;
+}
+
+#define PAGE_IN_SPRAM_DIE0(page) ((page_to_phys(page)>=0x59000000) && (page_to_phys(page)<0x59400000))
+#define PAGE_IN_SPRAM_DIE1(page) ((page_to_phys(page)>=0x79000000) && (page_to_phys(page)<0x79400000))
+static int do_vb_retrive_mem_node(struct dma_buf *dmabuf, int *nid)
+{
+ int ret = 0;
+ struct dma_buf_attachment *attach;
+ struct sg_table *sgt;
+ struct page *page = NULL;
+
+ get_dma_buf(dmabuf);
+ attach = dma_buf_attach(dmabuf, mmz_vb_dev);
+ if (IS_ERR(attach)) {
+ ret = PTR_ERR(attach);
+ /* put dmabuf back */
+ dma_buf_put(dmabuf);
+ return ret;
+ }
+
+ sgt = dma_buf_map_attachment(attach, DMA_BIDIRECTIONAL);
+ if (IS_ERR(sgt)) {
+ ret = PTR_ERR(sgt);
+ dma_buf_detach(dmabuf, attach);
+ dma_buf_put(dmabuf);
+ return ret;
+ }
+
+ page = sg_page(sgt->sgl);
+ if (unlikely(PAGE_IN_SPRAM_DIE0(page))) {
+ *nid = 0;
+ }
+ else if(unlikely(PAGE_IN_SPRAM_DIE1(page))) {
+ *nid = 1;
+ }
+ else
+ *nid = page_to_nid(page);
+
+ dma_buf_unmap_attachment(attach, sgt, DMA_BIDIRECTIONAL);
+ /* detach */
+ dma_buf_detach(dmabuf, attach);
+ /* put dmabuf back */
+ dma_buf_put(dmabuf);
+
+ pr_debug("%s, mem node is %d\n", __func__, *nid);
+ return ret;
+}
+
+static int vb_retrieve_mem_node(struct esVB_RETRIEVE_MEM_NODE_CMD_S *retrieveMemNodeCmd)
+{
+ int ret = 0;
+ struct dma_buf *dmabuf;
+ struct vm_area_struct *vma = NULL;
+ vm_flags_t vm_flags;
+ struct mm_struct *mm = current->mm;
+ u64 vaddr;
+
+ /* If cpu_vaddr is NULL, then try to retrieve mem node id by fd */
+ if (retrieveMemNodeCmd->cpu_vaddr == NULL) {
+ /* get dmabuf handle */
+ dmabuf = dma_buf_get(retrieveMemNodeCmd->fd);
+ if (IS_ERR(dmabuf)) {
+ return PTR_ERR(dmabuf);
+ }
+
+ ret = do_vb_retrive_mem_node(dmabuf, &retrieveMemNodeCmd->numa_node);
+ /* put dmabuf back */
+ dma_buf_put(dmabuf);
+ }
+ else {
+ vaddr = (u64)retrieveMemNodeCmd->cpu_vaddr;
+ mmap_read_lock(mm);
+ vma = vma_lookup(mm, vaddr & PAGE_MASK);
+ if (!vma) {
+ pr_err("Failed to vma_lookup!\n");
+ return -EFAULT;
+ }
+ vm_flags = vma->vm_flags;
+ mmap_read_unlock(mm);
+
+ if (!(vm_flags & (VM_IO | VM_PFNMAP)) || (NULL == vma->vm_private_data)) {
+ pr_debug("This vaddr is NOT mmapped with VM_PFNMAP!\n");
+ return -EFAULT;
+ }
+ dmabuf = vma->vm_private_data;
+ ret = do_vb_retrive_mem_node(dmabuf, &retrieveMemNodeCmd->numa_node);
+ }
+
+ return ret;
+}
+
+static int vb_get_dmabuf_size(struct esVB_DMABUF_SIZE_CMD_S *getDmabufSizeCmd)
+{
+ int ret = 0;
+ struct dma_buf *dmabuf;
+
+ /* get dmabuf handle */
+ dmabuf = dma_buf_get(getDmabufSizeCmd->fd);
+ if (IS_ERR(dmabuf)) {
+ return -EINVAL;
+ }
+
+ /* minus 1 because it was +1 by dma_buf_get */
+ getDmabufSizeCmd->size = dmabuf->size;
+
+ dma_buf_put(dmabuf);
+
+ return ret;
+}
+
+static int mmz_vb_init_memory_region(void)
+{
+ struct mmz_vb_priv *mmz_vb_priv = g_mmz_vb_priv;
+ struct esVB_K_MMZ_S *partitions = &mmz_vb_priv->partitions;
+ int nid, part = 0;
+ int partitionID = 0;
+ char blkName[BLOCK_MAX_NAME];
+ struct mem_block *memblock = NULL;
+
+ for (nid = 0; nid < 2; nid++) {
+ for (part = 0; part < 2; part++) {
+ snprintf(blkName, sizeof(blkName), "%s%d_part_%d", es_mmz_name_prefix, nid, part);
+ memblock = eswin_rsvmem_get_memblock(blkName);
+ if (memblock) {
+ partitions->mem_blocks[partitionID] = memblock;
+ dev_info(mmz_vb_dev, "%s was found successfully\n", blkName);
+ partitionID++;
+ }
+ else {
+ dev_dbg(mmz_vb_dev, "%s was NOT found\n", blkName);
+ }
+ }
+ }
+
+ /* Indicate how many VB memory block have been correctly initialized */
+ return partitionID;
+}
diff --git a/drivers/memory/eswin/es_proc/Kconfig b/drivers/memory/eswin/es_proc/Kconfig
new file mode 100644
index 000000000000..d1a2f8220e9b
--- /dev/null
+++ b/drivers/memory/eswin/es_proc/Kconfig
@@ -0,0 +1,7 @@
+# SPDX-License-Identifier: GPL-2.0
+
+config ESWIN_PROC
+ tristate "ESWIN MMZ proc interface"
+ help
+ MMZ proc interface for user space.
+
diff --git a/drivers/memory/eswin/es_proc/Makefile b/drivers/memory/eswin/es_proc/Makefile
new file mode 100644
index 000000000000..2afd49103e81
--- /dev/null
+++ b/drivers/memory/eswin/es_proc/Makefile
@@ -0,0 +1,7 @@
+# SPDX-License-Identifier: GPL-2.0
+obj-$(CONFIG_ESWIN_PROC) += es_proc.o
+ccflags-y := -DDEBUG
+
+ES_PROC_HEADER := drivers/memory/eswin/es_proc/include/linux
+
+COPY_HEADERS := $(shell cp $(ES_PROC_HEADER)/*.h include/linux)
diff --git a/drivers/memory/eswin/es_proc/es_proc.c b/drivers/memory/eswin/es_proc/es_proc.c
new file mode 100644
index 000000000000..b0eb521b247c
--- /dev/null
+++ b/drivers/memory/eswin/es_proc/es_proc.c
@@ -0,0 +1,233 @@
+// SPDX-License-Identifier: GPL-2.0-only
+/*
+ * ESWIN proc APIs for MMZ_VB
+ *
+ * Copyright 2024 Beijing ESWIN Computing Technology Co., Ltd.
+ * Authors:
+ * HuangYiFeng<huangyifeng@eswincomputing.com>
+ *
+ */
+
+#include <linux/module.h>
+#include <linux/kernel.h>
+#include <linux/printk.h>
+#include <linux/fs.h>
+#include <linux/seq_file.h>
+#include <linux/proc_fs.h>
+#include <linux/slab.h>
+#include <asm/uaccess.h>
+#include "include/linux/es_proc.h"
+
+static struct list_head list;
+static es_proc_entry_t *proc_entry = NULL;
+
+static int es_seq_show(struct seq_file *s, void *p)
+{
+ es_proc_entry_t *oldsentry = s->private;
+ es_proc_entry_t sentry;
+
+ if (oldsentry == NULL) {
+ pr_err("%s %d- parameter invalid!\n", __func__,__LINE__);
+ return -1;
+ }
+ memset(&sentry, 0, sizeof(es_proc_entry_t));
+ /* only these two parameters are used */
+ sentry.seqfile = s;
+ sentry.private = oldsentry->private;
+ oldsentry->read(&sentry);
+ return 0;
+}
+
+static ssize_t es_procwrite(struct file *file, const char __user *buf,
+ size_t count, loff_t *ppos)
+{
+ es_proc_entry_t *item = pde_data(file_inode(file));
+
+ if ((item != NULL) && (item->write != NULL)) {
+ return item->write(item, buf, count, (long long *)ppos);
+ }
+
+ return -ENOSYS;
+}
+
+static int es_procopen(struct inode *inode, struct file *file)
+{
+ es_proc_entry_t *sentry = pde_data(inode);
+
+ if ((sentry != NULL) && (sentry->open != NULL)) {
+ sentry->open(sentry);
+ }
+ return single_open(file, es_seq_show, sentry);
+}
+
+static const struct proc_ops es_proc_ops = {
+ .proc_open = es_procopen,
+ .proc_read = seq_read,
+ .proc_write = es_procwrite,
+ .proc_lseek = seq_lseek,
+ .proc_release = single_release
+};
+
+es_proc_entry_t *es_create_proc(const char *name, es_proc_entry_t *parent)
+{
+ struct proc_dir_entry *entry = NULL;
+ es_proc_entry_t *sentry = NULL;
+
+ sentry = kzalloc(sizeof(struct es_proc_dir_entry), GFP_KERNEL);
+ if (sentry == NULL) {
+ pr_err("%s %d - kmalloc failed!\n",__func__,__LINE__);
+ return NULL;
+ }
+
+ strncpy(sentry->name, name, sizeof(sentry->name) - 1);
+
+ if (parent == NULL) {
+ entry = proc_create_data(name, 0, NULL, &es_proc_ops, sentry);
+ } else {
+ entry = proc_create_data(name, 0, parent->proc_dir_entry, &es_proc_ops, sentry);
+ }
+ if (entry == NULL) {
+ pr_err("%s %d - create_proc_entry failed!\n",__func__,__LINE__);
+ kfree(sentry);
+ sentry = NULL;
+ return NULL;
+ }
+ sentry->proc_dir_entry = entry;
+ sentry->open = NULL;
+
+ list_add_tail(&(sentry->node), &list);
+ return sentry;
+}
+
+void es_remove_proc(const char *name, es_proc_entry_t *parent)
+{
+ struct es_proc_dir_entry *sproc = NULL;
+
+ if (name == NULL) {
+ pr_err("%s %d - parameter invalid!\n",__func__,__LINE__);
+ return;
+ }
+ if (parent != NULL) {
+ remove_proc_entry(name, parent->proc_dir_entry);
+ } else {
+ remove_proc_entry(name, NULL);
+ }
+ list_for_each_entry(sproc, &list, node) {
+ if (strncmp(sproc->name, name, sizeof(sproc->name)) == 0) {
+ list_del(&(sproc->node));
+ break;
+ }
+ }
+ if (sproc != NULL) {
+ kfree(sproc);
+ }
+}
+
+es_proc_entry_t *es_create_proc_entry(const char *name,
+ es_proc_entry_t *parent)
+{
+ parent = proc_entry;
+
+ return es_create_proc(name, parent);
+}
+EXPORT_SYMBOL(es_create_proc_entry);
+
+void es_remove_proc_entry(const char *name, es_proc_entry_t *parent)
+{
+ parent = proc_entry;
+ es_remove_proc(name, parent);
+ return;
+}
+EXPORT_SYMBOL(es_remove_proc_entry);
+
+es_proc_entry_t *es_proc_mkdir(const char *name, es_proc_entry_t *parent)
+{
+ struct proc_dir_entry *proc = NULL;
+ struct es_proc_dir_entry *sproc = NULL;
+
+ sproc = kzalloc(sizeof(struct es_proc_dir_entry), GFP_KERNEL);
+ if (sproc == NULL) {
+ pr_err("%s %d - kmalloc failed!\n",__func__,__LINE__);
+ return NULL;
+ }
+
+ strncpy(sproc->name, name, sizeof(sproc->name) - 1);
+
+ if (parent != NULL) {
+ proc = proc_mkdir_data(name, 0, parent->proc_dir_entry, sproc);
+ } else {
+ proc = proc_mkdir_data(name, 0, NULL, sproc);
+ }
+ if (proc == NULL) {
+ pr_err("%s %d - proc_mkdir failed!\n",__func__,__LINE__);
+ kfree(sproc);
+ sproc = NULL;
+ return NULL;
+ }
+ sproc->proc_dir_entry = proc;
+
+ list_add_tail(&(sproc->node), &list);
+ return sproc;
+}
+EXPORT_SYMBOL(es_proc_mkdir);
+
+void es_remove_proc_root(const char *name, es_proc_entry_t *parent)
+{
+ struct es_proc_dir_entry *sproc = NULL;
+
+ if (name == NULL) {
+ pr_err("%s %d - parameter invalid!\n",__func__,__LINE__);
+ return;
+ }
+ if (parent != NULL) {
+ remove_proc_entry(name, parent->proc_dir_entry);
+ } else {
+ remove_proc_entry(name, NULL);
+ }
+ list_for_each_entry(sproc, &list, node) {
+ if (strncmp(sproc->name, name, sizeof(sproc->name)) == 0) {
+ list_del(&(sproc->node));
+ break;
+ }
+ }
+ if (sproc != NULL) {
+ kfree(sproc);
+ }
+}
+
+int es_seq_printf(es_proc_entry_t *entry, const char *fmt, ...)
+{
+ struct seq_file *s = (struct seq_file *)(entry->seqfile);
+ va_list args;
+ int r = 0;
+
+ va_start(args, fmt);
+ seq_vprintf(s, fmt, args);
+ va_end(args);
+
+ return r;
+}
+EXPORT_SYMBOL(es_seq_printf);
+
+static int __init es_proc_init(void)
+{
+ INIT_LIST_HEAD(&list);
+ proc_entry = es_proc_mkdir("umap", NULL);
+ if (proc_entry == NULL) {
+ pr_err("init, proc mkdir error!\n");
+ return -EPERM;
+ }
+ return 0;
+}
+
+static void __exit es_proc_exit(void)
+{
+ es_remove_proc_root("umap", NULL);
+}
+
+module_init(es_proc_init);
+module_exit(es_proc_exit);
+
+MODULE_DESCRIPTION("ES Procfile Driver");
+MODULE_AUTHOR("huangyifeng@eswincomputing.com");
+MODULE_LICENSE("GPL v2");
diff --git a/drivers/memory/eswin/es_proc/include/linux/es_proc.h b/drivers/memory/eswin/es_proc/include/linux/es_proc.h
new file mode 100644
index 000000000000..65c73460e11b
--- /dev/null
+++ b/drivers/memory/eswin/es_proc/include/linux/es_proc.h
@@ -0,0 +1,40 @@
+// SPDX-License-Identifier: GPL-2.0-only
+/*
+ * Header file of es_proc.c
+ *
+ * Copyright 2024 Beijing ESWIN Computing Technology Co., Ltd.
+ * Authors:
+ * HuangYiFeng<huangyifeng@eswincomputing.com>
+ *
+ */
+
+#ifndef __ES_PROC__
+#define __ES_PROC__
+
+#define PROC_ENTRY_VI "vi"
+#define PROC_ENTRY_VO "vo"
+#define PROC_ENTRY_VB "vb"
+#define PROC_ENTRY_ISP "isp"
+
+// proc
+typedef struct es_proc_dir_entry {
+ char name[50];
+ void *proc_dir_entry;
+ int (*open)(struct es_proc_dir_entry *entry);
+ int (*read)(struct es_proc_dir_entry *entry);
+ int (*write)(struct es_proc_dir_entry *entry, const char *buf,
+ int count, long long *);
+ void *private;
+ void *seqfile;
+ struct list_head node;
+} es_proc_entry_t;
+
+extern es_proc_entry_t *es_create_proc_entry(const char *name,
+ es_proc_entry_t *parent);
+extern es_proc_entry_t *es_proc_mkdir(const char *name,
+ es_proc_entry_t *parent);
+extern void es_remove_proc_entry(const char *name, es_proc_entry_t *parent);
+extern int es_seq_printf(es_proc_entry_t *entry, const char *fmt, ...)
+ __attribute__((format(printf, 2, 3)));
+
+#endif
diff --git a/drivers/memory/eswin/es_rsvmem_heap/Kconfig b/drivers/memory/eswin/es_rsvmem_heap/Kconfig
new file mode 100644
index 000000000000..3b425da42e2f
--- /dev/null
+++ b/drivers/memory/eswin/es_rsvmem_heap/Kconfig
@@ -0,0 +1,6 @@
+# SPDX-License-Identifier: GPL-2.0
+
+config ESWIN_RSVMEM_HEAP
+ tristate "ESWIN reserved memory heap"
+ help
+ ESWIN reserved memory heap device.
diff --git a/drivers/memory/eswin/es_rsvmem_heap/Makefile b/drivers/memory/eswin/es_rsvmem_heap/Makefile
new file mode 100644
index 000000000000..e468c514bb7b
--- /dev/null
+++ b/drivers/memory/eswin/es_rsvmem_heap/Makefile
@@ -0,0 +1,8 @@
+# SPDX-License-Identifier: GPL-2.0
+obj-$(CONFIG_ESWIN_RSVMEM_HEAP) += eswin_rsvmem_heap.o eswin_rsvmem_common.o dmabuf-heap-import-helper.o
+
+
+ES_RSVMEM_HEADER := drivers/memory/eswin/es_rsvmem_heap/include/linux
+
+COPY_HEADERS := $(shell cp $(ES_RSVMEM_HEADER)/*.h include/linux)
+
diff --git a/drivers/memory/eswin/es_rsvmem_heap/dmabuf-heap-import-helper.c b/drivers/memory/eswin/es_rsvmem_heap/dmabuf-heap-import-helper.c
new file mode 100644
index 000000000000..fdbcfe7e6c70
--- /dev/null
+++ b/drivers/memory/eswin/es_rsvmem_heap/dmabuf-heap-import-helper.c
@@ -0,0 +1,652 @@
+// SPDX-License-Identifier: GPL-2.0-only
+/*
+ * ESWIN DMABUF heap helper APIs
+ *
+ * Copyright 2024 Beijing ESWIN Computing Technology Co., Ltd.
+ * Authors:
+ * LinMin<linmin@eswincomputing.com>
+ *
+ */
+
+#include <linux/export.h>
+#include <linux/slab.h>
+#include <linux/errno.h>
+#include <linux/fcntl.h>
+#include <linux/scatterlist.h>
+#include <linux/dma-heap.h>
+#include <linux/dmabuf-heap-import-helper.h>
+
+struct drm_prime_member {
+ struct dma_buf *dma_buf;
+ uint64_t handle;
+
+ struct rb_node dmabuf_rb;
+ struct rb_node handle_rb;
+};
+
+static int dmabuf_heap_add_buf_handle(struct dmaheap_file_private *prime_fpriv,
+ struct dma_buf *dma_buf, uint64_t handle)
+{
+ struct drm_prime_member *member;
+ struct rb_node **p, *rb;
+
+ member = kmalloc(sizeof(*member), GFP_KERNEL);
+ if (!member)
+ return -ENOMEM;
+
+ get_dma_buf(dma_buf);
+ member->dma_buf = dma_buf;
+ member->handle = handle;
+
+ rb = NULL;
+ p = &prime_fpriv->dmabufs.rb_node;
+ while (*p) {
+ struct drm_prime_member *pos;
+
+ rb = *p;
+ pos = rb_entry(rb, struct drm_prime_member, dmabuf_rb);
+ if (dma_buf > pos->dma_buf)
+ p = &rb->rb_right;
+ else
+ p = &rb->rb_left;
+ }
+ rb_link_node(&member->dmabuf_rb, rb, p);
+ rb_insert_color(&member->dmabuf_rb, &prime_fpriv->dmabufs);
+
+ rb = NULL;
+ p = &prime_fpriv->handles.rb_node;
+ while (*p) {
+ struct drm_prime_member *pos;
+
+ rb = *p;
+ pos = rb_entry(rb, struct drm_prime_member, handle_rb);
+ if (handle > pos->handle)
+ p = &rb->rb_right;
+ else
+ p = &rb->rb_left;
+ }
+ rb_link_node(&member->handle_rb, rb, p);
+ rb_insert_color(&member->handle_rb, &prime_fpriv->handles);
+
+ return 0;
+}
+
+static int dmabuf_heap_lookup_buf_handle(struct dmaheap_file_private *prime_fpriv,
+ struct dma_buf *dma_buf,
+ uint64_t *handle)
+{
+ struct rb_node *rb;
+
+ rb = prime_fpriv->dmabufs.rb_node;
+ while (rb) {
+ struct drm_prime_member *member;
+
+ member = rb_entry(rb, struct drm_prime_member, dmabuf_rb);
+ if (member->dma_buf == dma_buf) {
+ *handle = member->handle;
+ return 0;
+ } else if (member->dma_buf < dma_buf) {
+ rb = rb->rb_right;
+ } else {
+ rb = rb->rb_left;
+ }
+ }
+
+ return -ENOENT;
+}
+
+static void _dmabuf_heap_remove_buf_handle(struct dmaheap_file_private *prime_fpriv,
+ struct dma_buf *dma_buf)
+{
+ struct rb_node *rb;
+
+ rb = prime_fpriv->dmabufs.rb_node;
+ while (rb) {
+ struct drm_prime_member *member;
+
+ member = rb_entry(rb, struct drm_prime_member, dmabuf_rb);
+ if (member->dma_buf == dma_buf) {
+ rb_erase(&member->handle_rb, &prime_fpriv->handles);
+ rb_erase(&member->dmabuf_rb, &prime_fpriv->dmabufs);
+
+ dma_buf_put(dma_buf);
+ kfree(member);
+ return;
+ } else if (member->dma_buf < dma_buf) {
+ rb = rb->rb_right;
+ } else {
+ rb = rb->rb_left;
+ }
+ }
+}
+
+void common_dmabuf_heap_import_init(struct heap_root *root, struct device *dev)
+{
+ memset(root, 0, sizeof(*root));
+
+ mutex_init(&root->lock);
+ INIT_LIST_HEAD(&root->header);
+
+ root->dev = dev;
+}
+EXPORT_SYMBOL(common_dmabuf_heap_import_init);
+
+void common_dmabuf_heap_import_uninit(struct heap_root *root)
+{
+ struct heap_mem *h, *tmp;
+
+ list_for_each_entry_safe(h, tmp, &root->header, list) {
+ common_dmabuf_heap_release(h);
+ }
+}
+EXPORT_SYMBOL(common_dmabuf_heap_import_uninit);
+
+static struct heap_mem *dmabuf_heap_import(struct heap_root *root, int fd)
+{
+ struct dma_buf *dma_buf;
+ struct dma_buf_attachment *attach;
+ struct sg_table *sgt;
+
+ uint64_t handle;
+ struct heap_mem *heap_obj;
+ int ret;
+
+ /* get dmabuf handle */
+ dma_buf = dma_buf_get(fd);
+ if (IS_ERR(dma_buf))
+ return ERR_CAST(dma_buf);
+
+ mutex_lock(&root->lock);
+
+ ret = dmabuf_heap_lookup_buf_handle(&root->fp, dma_buf, &handle);
+ if (ret == 0) {
+ heap_obj = (struct heap_mem *)handle;
+ dma_buf_put(dma_buf);
+ kref_get(&heap_obj->refcount);
+ mutex_unlock(&root->lock);
+ return heap_obj;
+ }
+
+ heap_obj = kzalloc(sizeof(*heap_obj), GFP_KERNEL);
+ if (!heap_obj) {
+ mutex_unlock(&root->lock);
+ dma_buf_put(dma_buf);
+ return ERR_PTR(-ENOMEM);
+ }
+
+ attach = dma_buf_attach(dma_buf, root->dev);
+ if (IS_ERR(attach)) {
+ ret = PTR_ERR(attach);
+ goto clean_up;
+ }
+
+ sgt = dma_buf_map_attachment(attach, DMA_BIDIRECTIONAL);
+ if (IS_ERR(sgt)) {
+ ret = PTR_ERR(sgt);
+ goto fail_detach;
+ }
+
+ heap_obj->dbuf_fd = fd;
+ heap_obj->dbuf = dma_buf;
+
+ heap_obj->import_attach = attach;
+ heap_obj->sgt = sgt;
+
+ heap_obj->root = root;
+ heap_obj->vaddr = NULL;
+ heap_obj->dir = DMA_BIDIRECTIONAL;
+
+ /* get_dma_buf was called in dmabuf_heap_add_buf_handle()*/
+ ret = dmabuf_heap_add_buf_handle(&root->fp, dma_buf, (uint64_t)heap_obj);
+ if (ret) {
+ goto fail_add_handle;
+ }
+
+ kref_init(&heap_obj->refcount);
+
+ list_add(&heap_obj->list, &root->header);
+
+ mutex_unlock(&root->lock);
+
+ dma_buf_put(dma_buf);
+
+ return heap_obj;
+
+fail_add_handle:
+fail_detach:
+ dma_buf_detach(dma_buf, attach);
+clean_up:
+ kfree(heap_obj);
+ mutex_unlock(&root->lock);
+ dma_buf_put(dma_buf);
+
+ return ERR_PTR(ret);
+}
+
+static struct heap_mem *dmabuf_heap_import_with_dma_buf_st(struct heap_root *root, struct dma_buf *dma_buf)
+{
+ struct dma_buf_attachment *attach;
+ struct sg_table *sgt;
+
+ uint64_t handle;
+ struct heap_mem *heap_obj;
+ int ret;
+
+ mutex_lock(&root->lock);
+
+ ret = dmabuf_heap_lookup_buf_handle(&root->fp, dma_buf, &handle);
+ if (ret == 0) {
+ heap_obj = (struct heap_mem *)handle;
+ kref_get(&heap_obj->refcount);
+ mutex_unlock(&root->lock);
+ return heap_obj;
+ }
+
+ heap_obj = kzalloc(sizeof(*heap_obj), GFP_KERNEL);
+ if (!heap_obj) {
+ mutex_unlock(&root->lock);
+ return ERR_PTR(-ENOMEM);
+ }
+
+ attach = dma_buf_attach(dma_buf, root->dev);
+ if (IS_ERR(attach)) {
+ ret = PTR_ERR(attach);
+ goto clean_up;
+ }
+
+ sgt = dma_buf_map_attachment(attach, DMA_BIDIRECTIONAL);
+ if (IS_ERR(sgt)) {
+ ret = PTR_ERR(sgt);
+ goto fail_detach;
+ }
+
+ heap_obj->dbuf_fd = -1;
+ heap_obj->dbuf = dma_buf;
+
+ heap_obj->import_attach = attach;
+ heap_obj->sgt = sgt;
+
+ heap_obj->root = root;
+ heap_obj->vaddr = NULL;
+ heap_obj->dir = DMA_BIDIRECTIONAL;
+
+ /* get_dma_buf was called in dmabuf_heap_add_buf_handle()*/
+ ret = dmabuf_heap_add_buf_handle(&root->fp, dma_buf, (uint64_t)heap_obj);
+ if (ret) {
+ goto fail_add_handle;
+ }
+
+ kref_init(&heap_obj->refcount);
+
+ list_add(&heap_obj->list, &root->header);
+
+ mutex_unlock(&root->lock);
+
+ return heap_obj;
+
+fail_add_handle:
+fail_detach:
+ dma_buf_detach(dma_buf, attach);
+clean_up:
+ kfree(heap_obj);
+ mutex_unlock(&root->lock);
+
+ return ERR_PTR(ret);
+}
+
+struct heap_mem *common_dmabuf_lookup_heapobj_by_fd(struct heap_root *root, int fd)
+{
+ int ret = 0;
+ struct dma_buf *dma_buf;
+ struct heap_mem *heap_obj;
+
+ /* get dmabuf handle */
+ dma_buf = dma_buf_get(fd);
+ if (IS_ERR(dma_buf))
+ return NULL;
+
+ mutex_lock(&root->lock);
+ ret = dmabuf_heap_lookup_buf_handle(&root->fp, dma_buf, (uint64_t *)&heap_obj);
+ mutex_unlock(&root->lock);
+
+ dma_buf_put(dma_buf);
+ if (0 == ret)
+ return heap_obj;
+ else
+ return NULL;
+}
+EXPORT_SYMBOL(common_dmabuf_lookup_heapobj_by_fd);
+
+struct heap_mem *common_dmabuf_lookup_heapobj_by_dma_buf_st(struct heap_root *root, struct dma_buf *dma_buf)
+{
+ int ret = 0;
+ struct heap_mem *heap_obj;
+
+ pr_debug("%s:dma_buf=0x%px, file_count=%ld\n",
+ __func__, dma_buf, file_count(dma_buf->file));
+ mutex_lock(&root->lock);
+ ret = dmabuf_heap_lookup_buf_handle(&root->fp, dma_buf, (uint64_t *)&heap_obj);
+ mutex_unlock(&root->lock);
+
+ if (0 == ret)
+ return heap_obj;
+ else
+ return NULL;
+}
+EXPORT_SYMBOL(common_dmabuf_lookup_heapobj_by_dma_buf_st);
+
+struct heap_mem *common_dmabuf_heap_import_from_user(struct heap_root *root, int fd)
+{
+ return dmabuf_heap_import(root, fd);
+}
+EXPORT_SYMBOL(common_dmabuf_heap_import_from_user);
+
+struct heap_mem *common_dmabuf_heap_import_from_user_with_dma_buf_st(struct heap_root *root, struct dma_buf *dma_buf)
+{
+ return dmabuf_heap_import_with_dma_buf_st(root, dma_buf);
+}
+EXPORT_SYMBOL(common_dmabuf_heap_import_from_user_with_dma_buf_st);
+
+static void __common_dmabuf_heap_release(struct kref *kref)
+{
+ struct heap_root *root;
+ struct heap_mem *heap_obj = container_of(kref, struct heap_mem, refcount);
+
+ WARN_ON(!heap_obj);
+ if (!heap_obj)
+ return;
+
+ root = heap_obj->root;
+ WARN_ON(!mutex_is_locked(&root->lock));
+ list_del(&heap_obj->list);
+
+ common_dmabuf_heap_umap_vaddr(heap_obj);
+
+ dma_buf_unmap_attachment(heap_obj->import_attach, heap_obj->sgt, heap_obj->dir);
+
+ dma_buf_detach(heap_obj->dbuf, heap_obj->import_attach);
+
+ /* dma_buf_put was called in _dmabuf_heap_remove_buf_handle()*/
+ _dmabuf_heap_remove_buf_handle(&root->fp, heap_obj->dbuf);
+
+ kfree(heap_obj);
+}
+
+void common_dmabuf_heap_release(struct heap_mem *heap_obj)
+{
+ struct heap_root *root = heap_obj->root;
+
+ mutex_lock(&root->lock);
+ kref_put(&heap_obj->refcount, __common_dmabuf_heap_release);
+ mutex_unlock(&root->lock);
+}
+EXPORT_SYMBOL(common_dmabuf_heap_release);
+
+void *common_dmabuf_heap_map_vaddr(struct heap_mem *heap_obj)
+{
+ struct dma_buf_map map;
+ int ret;
+
+ WARN_ON(!heap_obj);
+ if (!heap_obj)
+ return NULL;
+
+ if (heap_obj->vaddr)
+ return heap_obj->vaddr;
+
+ ret = dma_buf_vmap(heap_obj->dbuf, &map);
+ if (ret)
+ return NULL;
+
+ WARN_ON_ONCE(map.is_iomem);
+ heap_obj->vaddr = map.vaddr;
+
+ return heap_obj->vaddr;
+}
+EXPORT_SYMBOL(common_dmabuf_heap_map_vaddr);
+
+void common_dmabuf_heap_umap_vaddr(struct heap_mem *heap_obj)
+{
+ struct dma_buf_map map;
+
+ WARN_ON(!heap_obj);
+ if (heap_obj && heap_obj->vaddr) {
+ map.vaddr = heap_obj->vaddr;
+ map.is_iomem = 0;
+ dma_buf_vunmap(heap_obj->dbuf, &map);
+ heap_obj->vaddr = NULL;
+ }
+}
+EXPORT_SYMBOL(common_dmabuf_heap_umap_vaddr);
+
+struct heap_mem *
+common_dmabuf_heap_import_from_kernel(struct heap_root *root, char *name, size_t len, unsigned int fd_flags)
+{
+ int dbuf_fd;
+
+ dbuf_fd = eswin_heap_kalloc(name, len, O_RDWR | fd_flags, 0);
+ if (dbuf_fd < 0) {
+ return ERR_PTR(dbuf_fd);
+ }
+
+ return dmabuf_heap_import(root, dbuf_fd);
+}
+EXPORT_SYMBOL(common_dmabuf_heap_import_from_kernel);
+
+struct esw_exp_attachment {
+ struct heap_mem *hmem;
+ struct sg_table table;
+ struct device *dev;
+ struct heap_root root;
+};
+
+// #define PRINT_ORIGINAL_SPLITTERS 1
+static int esw_common_heap_attach(struct dma_buf *dmabuf,
+ struct dma_buf_attachment *attachment)
+{
+ struct esw_export_buffer_info *buffer = dmabuf->priv;
+ struct esw_exp_attachment *a;
+ int out_mapped_nents[1];
+ int ret = 0;
+ struct sg_table *sgt = NULL;
+ struct scatterlist *sg;
+ int i;
+ size_t size, len;
+
+ a = kzalloc(sizeof(*a), GFP_KERNEL);
+ if (!a)
+ return -ENOMEM;
+
+ a->dev = attachment->dev;
+
+ common_dmabuf_heap_import_init(&a->root, a->dev);
+ a->hmem = common_dmabuf_heap_import_from_user(&a->root, buffer->dbuf_fd);
+ if (IS_ERR(a->hmem))
+ return PTR_ERR(a->hmem);
+
+ ret = sg_split(a->hmem->sgt->sgl, a->hmem->sgt->nents, buffer->slice.offset, 1, &buffer->slice.len,
+ &a->table.sgl, &out_mapped_nents[0], GFP_KERNEL);
+ if (ret) {
+ common_dmabuf_heap_release(a->hmem);
+ kfree(a);
+ return ret;
+ }
+ a->table.nents = out_mapped_nents[0];
+ a->table.orig_nents = out_mapped_nents[0];
+ sgt = &a->table;
+ #ifdef PRINT_ORIGINAL_SPLITTERS
+ {
+ pr_info("%s:orig:sgt->orig_nents=%d, out_mapped_nents[0]=%d\n",
+ __func__, sgt->orig_nents, out_mapped_nents[0]);
+ for_each_sg(sgt->sgl, sg, sgt->orig_nents, i) {
+ pr_info("orig[%d]:sg->length=0x%x, sg_dma_len=0x%x, sg_phys=0x%lx\n",
+ i, sg->length, sg_dma_len(sg), (unsigned long)sg_phys(sg));
+ }
+ }
+ #endif
+ /* Re-format the splitted sg list in the actual slice len */
+ {
+ size = buffer->slice.len;
+ for_each_sg(sgt->sgl, sg, sgt->orig_nents, i) {
+ if (sg->length >= size) {
+ sg->length = size;
+ sg_dma_len(sg) = size;
+ sg_mark_end(sg);
+ pr_debug("refmt[%d]:sg->length=0x%x, sg_dma_len=0x%x, sg_phys=0x%lx\n",
+ i, sg->length, sg_dma_len(sg), (unsigned long)sg_phys(sg));
+ break;
+ }
+ len = min_t(size_t, size, sg->length);
+ size -= len;
+ pr_debug("refmt[%d]:sg->length=0x%x, sg_dma_len=0x%x, sg_phys=0x%lx\n",
+ i, sg->length, sg_dma_len(sg), (unsigned long)sg_phys(sg));
+ }
+ sgt->orig_nents = sgt->nents = i + 1;
+ }
+
+ attachment->priv = a;
+
+ return ret;
+}
+
+static void esw_common_heap_detach(struct dma_buf *dmabuf,
+ struct dma_buf_attachment *attachment)
+{
+ struct esw_exp_attachment *a = attachment->priv;
+
+ kfree(a->table.sgl);
+ common_dmabuf_heap_release(a->hmem);
+ common_dmabuf_heap_import_uninit(&a->root);
+ kfree(a);
+}
+
+static struct sg_table *esw_common_map_dma_buf(struct dma_buf_attachment *attachment,
+ enum dma_data_direction direction)
+{
+ struct esw_exp_attachment *a = attachment->priv;
+ return &a->table;
+}
+
+static void esw_common_unmap_dma_buf(struct dma_buf_attachment *attachment,
+ struct sg_table *table,
+ enum dma_data_direction direction)
+{
+}
+
+static int esw_common_mmap(struct dma_buf *dmabuf, struct vm_area_struct *vma)
+{
+ struct esw_export_buffer_info *buffer = dmabuf->priv;
+ // printk("%s enter\n", __func__);
+ return dma_buf_mmap(buffer->dmabuf, vma, buffer->slice.offset >> PAGE_SHIFT);
+}
+
+static int esw_common_dma_buf_begin_cpu_access(struct dma_buf *dmabuf,
+ enum dma_data_direction direction)
+{
+ struct esw_export_buffer_info *buffer = dmabuf->priv;
+ return dma_buf_begin_cpu_access(buffer->dmabuf, direction);
+}
+
+static int esw_common_dma_buf_end_cpu_access(struct dma_buf *dmabuf,
+ enum dma_data_direction direction)
+{
+ struct esw_export_buffer_info *buffer = dmabuf->priv;
+ return dma_buf_end_cpu_access(buffer->dmabuf, direction);
+}
+
+static int esw_common_heap_vmap(struct dma_buf *dmabuf, struct dma_buf_map *map)
+{
+ struct esw_export_buffer_info *buffer = dmabuf->priv;
+ struct dma_buf_map pmap;
+ int ret;
+
+ ret = dma_buf_vmap(buffer->dmabuf, &pmap);
+
+ map->is_iomem = false;
+ map->vaddr_iomem = pmap.vaddr_iomem + buffer->slice.offset;
+
+ return ret;
+}
+
+static void esw_common_heap_vunmap(struct dma_buf *dmabuf, struct dma_buf_map *map)
+{
+ struct esw_export_buffer_info *buffer = dmabuf->priv;
+ struct dma_buf_map pmap = *map;
+
+ pmap.vaddr_iomem -= buffer->slice.offset;
+ dma_buf_vunmap(buffer->dmabuf, &pmap);
+}
+
+static void esw_common_dma_buf_release(struct dma_buf *dmabuf)
+{
+ struct esw_export_buffer_info *buffer = dmabuf->priv;
+
+ // printk("%s %d\n", __func__, __LINE__);
+
+ dma_buf_put(buffer->dmabuf);
+ kfree(buffer);
+}
+
+static const struct dma_buf_ops esw_common_buf_ops = {
+ .attach = esw_common_heap_attach,
+ .detach = esw_common_heap_detach,
+ .map_dma_buf = esw_common_map_dma_buf,
+ .unmap_dma_buf = esw_common_unmap_dma_buf,
+ .begin_cpu_access = esw_common_dma_buf_begin_cpu_access,
+ .end_cpu_access = esw_common_dma_buf_end_cpu_access,
+ .mmap = esw_common_mmap,
+ .vmap = esw_common_heap_vmap,
+ .vunmap = esw_common_heap_vunmap,
+ .release = esw_common_dma_buf_release,
+};
+
+int esw_common_dmabuf_split_export(int dbuf_fd, unsigned int offset, size_t len, int fd_flags, char *name)
+{
+ struct esw_export_buffer_info *buffer_info;
+ DEFINE_DMA_BUF_EXPORT_INFO(exp_info);
+ int fd;
+ struct dma_buf *dmabuf;
+
+ buffer_info = kzalloc(sizeof(*buffer_info), GFP_KERNEL);
+ if (!buffer_info)
+ return -ENOMEM;
+
+ buffer_info->dbuf_fd = dbuf_fd;
+ buffer_info->fd_flags = fd_flags;
+ buffer_info->slice.offset = offset;
+ buffer_info->slice.len = len;
+ snprintf(buffer_info->name, sizeof(buffer_info->name), "%s", name);
+
+ buffer_info->dmabuf = dma_buf_get(buffer_info->dbuf_fd);
+ if (IS_ERR(buffer_info->dmabuf))
+ return PTR_ERR(buffer_info->dmabuf);
+
+// printk("input slice: oft=0x%d, len=%lu\n", buffer_info->slice.offset, buffer_info->slice.len);
+
+ buffer_info->slice.offset = PAGE_ALIGN(buffer_info->slice.offset);
+ buffer_info->slice.len = PAGE_ALIGN(buffer_info->slice.len);
+
+// printk("align slice: oft=0x%d, len=%lu\n", buffer_info->slice.offset, buffer_info->slice.len);
+
+ /* create the dmabuf */
+ exp_info.exp_name = buffer_info->name;
+ exp_info.ops = &esw_common_buf_ops;
+ exp_info.size = buffer_info->slice.len;
+ exp_info.flags = buffer_info->fd_flags;
+ exp_info.priv = buffer_info;
+
+ dmabuf = dma_buf_export(&exp_info);
+ if (IS_ERR(dmabuf)) {
+ return PTR_ERR(dmabuf);
+ }
+
+ fd = dma_buf_fd(dmabuf, buffer_info->fd_flags);
+ if (fd < 0) {
+ dma_buf_put(dmabuf);
+ /* put the splitted dmabuf, the esw_common_dma_buf_release will be called,
+ the parent dmabuf will be put and the buffer_info will be free at that time */
+ }
+ return fd;
+}
+
+EXPORT_SYMBOL(esw_common_dmabuf_split_export);
diff --git a/drivers/memory/eswin/es_rsvmem_heap/eswin_rsvmem_common.c b/drivers/memory/eswin/es_rsvmem_heap/eswin_rsvmem_common.c
new file mode 100644
index 000000000000..067f30627a46
--- /dev/null
+++ b/drivers/memory/eswin/es_rsvmem_heap/eswin_rsvmem_common.c
@@ -0,0 +1,447 @@
+// SPDX-License-Identifier: GPL-2.0-only
+/*
+ * ESWIN heap APIs
+ *
+ * Copyright 2024 Beijing ESWIN Computing Technology Co., Ltd.
+ * Authors:
+ * LinMin<linmin@eswincomputing.com>
+ *
+ */
+
+#include <linux/cdev.h>
+#include <linux/debugfs.h>
+#include <linux/device.h>
+#include <linux/dma-buf.h>
+#include <linux/err.h>
+#include <linux/xarray.h>
+#include <linux/list.h>
+#include <linux/slab.h>
+#include <linux/nospec.h>
+#include <linux/uaccess.h>
+#include <linux/syscalls.h>
+#include <linux/eswin_rsvmem_common.h>
+#include "include/uapi/linux/eswin_rsvmem_common.h"
+
+#define DEVNAME "eswin_heap"
+
+#define NUM_HEAP_MINORS 128
+
+/**
+ * struct eswin_heap - represents a dmabuf heap in the system
+ * @name: used for debugging/device-node name
+ * @ops: ops struct for this heap
+ * @heap_devt heap device node
+ * @list list head connecting to list of heaps
+ * @heap_cdev heap char device
+ *
+ * Represents a heap of memory from which buffers can be made.
+ */
+struct eswin_heap {
+ const char *name;
+ const struct eswin_heap_ops *ops;
+ void *priv;
+ dev_t heap_devt;
+ struct list_head list;
+ struct cdev heap_cdev;
+};
+
+static LIST_HEAD(heap_list);
+static DEFINE_MUTEX(heap_list_lock);
+static dev_t eswin_heap_devt;
+static struct class *eswin_heap_class;
+static DEFINE_XARRAY_ALLOC(eswin_heap_minors);
+
+static int eswin_heap_buffer_alloc(struct eswin_heap *heap, size_t len,
+ unsigned int fd_flags,
+ unsigned int heap_flags)
+{
+ struct dma_buf *dmabuf;
+ int fd;
+
+ /*
+ * Allocations from all heaps have to begin
+ * and end on page boundaries.
+ */
+ len = PAGE_ALIGN(len);
+ if (!len)
+ return -EINVAL;
+
+ dmabuf = heap->ops->allocate(heap, len, fd_flags, heap_flags);
+ if (IS_ERR(dmabuf))
+ return PTR_ERR(dmabuf);
+
+ fd = dma_buf_fd(dmabuf, fd_flags);
+ if (fd < 0) {
+ dma_buf_put(dmabuf);
+ /* just return, as put will call release and that will free */
+ }
+ return fd;
+}
+
+static int eswin_heap_open(struct inode *inode, struct file *file)
+{
+ struct eswin_heap *heap;
+
+ heap = xa_load(&eswin_heap_minors, iminor(inode));
+ if (!heap) {
+ pr_err("eswin_heap: minor %d unknown.\n", iminor(inode));
+ return -ENODEV;
+ }
+
+ /* instance data as context */
+ file->private_data = heap;
+ nonseekable_open(inode, file);
+
+ return 0;
+}
+
+static long eswin_heap_ioctl_allocate(struct file *file, void *data)
+{
+ struct eswin_heap_allocation_data *heap_allocation = data;
+ struct eswin_heap *heap = file->private_data;
+ int fd;
+
+ if (heap_allocation->fd)
+ return -EINVAL;
+
+ if (heap_allocation->fd_flags & ~ESWIN_HEAP_VALID_FD_FLAGS)
+ return -EINVAL;
+
+ if (heap_allocation->heap_flags & ~ESWIN_HEAP_VALID_HEAP_FLAGS)
+ return -EINVAL;
+
+ fd = eswin_heap_buffer_alloc(heap, heap_allocation->len,
+ heap_allocation->fd_flags,
+ heap_allocation->heap_flags);
+ if (fd < 0)
+ return fd;
+
+ heap_allocation->fd = fd;
+
+ return 0;
+}
+
+static unsigned int eswin_heap_ioctl_cmds[] = {
+ ESWIN_HEAP_IOCTL_ALLOC,
+};
+
+static long eswin_heap_ioctl(struct file *file, unsigned int ucmd,
+ unsigned long arg)
+{
+ char stack_kdata[128];
+ char *kdata = stack_kdata;
+ unsigned int kcmd;
+ unsigned int in_size, out_size, drv_size, ksize;
+ int nr = _IOC_NR(ucmd);
+ int ret = 0;
+
+ if (nr >= ARRAY_SIZE(eswin_heap_ioctl_cmds))
+ return -EINVAL;
+
+ nr = array_index_nospec(nr, ARRAY_SIZE(eswin_heap_ioctl_cmds));
+ /* Get the kernel ioctl cmd that matches */
+ kcmd = eswin_heap_ioctl_cmds[nr];
+
+ /* Figure out the delta between user cmd size and kernel cmd size */
+ drv_size = _IOC_SIZE(kcmd);
+ out_size = _IOC_SIZE(ucmd);
+ in_size = out_size;
+ if ((ucmd & kcmd & IOC_IN) == 0)
+ in_size = 0;
+ if ((ucmd & kcmd & IOC_OUT) == 0)
+ out_size = 0;
+ ksize = max(max(in_size, out_size), drv_size);
+
+ /* If necessary, allocate buffer for ioctl argument */
+ if (ksize > sizeof(stack_kdata)) {
+ kdata = kmalloc(ksize, GFP_KERNEL);
+ if (!kdata)
+ return -ENOMEM;
+ }
+
+ if (copy_from_user(kdata, (void __user *)arg, in_size) != 0) {
+ ret = -EFAULT;
+ goto err;
+ }
+
+ /* zero out any difference between the kernel/user structure size */
+ if (ksize > in_size)
+ memset(kdata + in_size, 0, ksize - in_size);
+
+ switch (kcmd) {
+ case ESWIN_HEAP_IOCTL_ALLOC:
+ ret = eswin_heap_ioctl_allocate(file, kdata);
+ break;
+ default:
+ ret = -ENOTTY;
+ goto err;
+ }
+
+ if (copy_to_user((void __user *)arg, kdata, out_size) != 0)
+ ret = -EFAULT;
+err:
+ if (kdata != stack_kdata)
+ kfree(kdata);
+ return ret;
+}
+
+static const struct file_operations eswin_heap_fops = {
+ .owner = THIS_MODULE,
+ .open = eswin_heap_open,
+ .unlocked_ioctl = eswin_heap_ioctl,
+#ifdef CONFIG_COMPAT
+ .compat_ioctl = eswin_heap_ioctl,
+#endif
+};
+
+/**
+ * eswin_heap_get_drvdata() - get per-subdriver data for the heap
+ * @heap: DMA-Heap to retrieve private data for
+ *
+ * Returns:
+ * The per-subdriver data for the heap.
+ */
+void *eswin_heap_get_drvdata(struct eswin_heap *heap)
+{
+ return heap->priv;
+}
+
+/**
+ * eswin_heap_get_name() - get heap name
+ * @heap: DMA-Heap to retrieve private data for
+ *
+ * Returns:
+ * The char* for the heap name.
+ */
+const char *eswin_heap_get_name(struct eswin_heap *heap)
+{
+ return heap->name;
+}
+
+struct eswin_heap *eswin_heap_add(const struct eswin_heap_export_info *exp_info)
+{
+ struct eswin_heap *heap, *h = NULL, *err_ret;
+ struct device *dev_ret;
+ unsigned int minor;
+ int ret;
+
+ if (!exp_info->name || !strcmp(exp_info->name, "")) {
+ pr_err("eswin_heap: Cannot add heap without a name\n");
+ return ERR_PTR(-EINVAL);
+ }
+
+ if (!exp_info->ops || !exp_info->ops->allocate) {
+ pr_err("eswin_heap: Cannot add heap with invalid ops struct\n");
+ return ERR_PTR(-EINVAL);
+ }
+
+ /* check the name is unique */
+ mutex_lock(&heap_list_lock);
+ list_for_each_entry(h, &heap_list, list) {
+ if (!strcmp(h->name, exp_info->name)) {
+ mutex_unlock(&heap_list_lock);
+ pr_err("eswin_heap: Already registered heap named %s\n",
+ exp_info->name);
+ return ERR_PTR(-EINVAL);
+ }
+ }
+ mutex_unlock(&heap_list_lock);
+
+ heap = kzalloc(sizeof(*heap), GFP_KERNEL);
+ if (!heap)
+ return ERR_PTR(-ENOMEM);
+
+ heap->name = exp_info->name;
+ heap->ops = exp_info->ops;
+ heap->priv = exp_info->priv;
+
+ /* Find unused minor number */
+ ret = xa_alloc(&eswin_heap_minors, &minor, heap,
+ XA_LIMIT(0, NUM_HEAP_MINORS - 1), GFP_KERNEL);
+ if (ret < 0) {
+ pr_err("eswin_heap: Unable to get minor number for heap\n");
+ err_ret = ERR_PTR(ret);
+ goto err0;
+ }
+
+ /* Create device */
+ heap->heap_devt = MKDEV(MAJOR(eswin_heap_devt), minor);
+
+ cdev_init(&heap->heap_cdev, &eswin_heap_fops);
+ ret = cdev_add(&heap->heap_cdev, heap->heap_devt, 1);
+ if (ret < 0) {
+ pr_err("eswin_heap: Unable to add char device\n");
+ err_ret = ERR_PTR(ret);
+ goto err1;
+ }
+
+ dev_ret = device_create(eswin_heap_class,
+ NULL,
+ heap->heap_devt,
+ NULL,
+ heap->name);
+ if (IS_ERR(dev_ret)) {
+ pr_err("eswin_heap: Unable to create device\n");
+ err_ret = ERR_CAST(dev_ret);
+ goto err2;
+ }
+ /* Add heap to the list */
+ mutex_lock(&heap_list_lock);
+ list_add(&heap->list, &heap_list);
+ mutex_unlock(&heap_list_lock);
+
+ return heap;
+
+err2:
+ cdev_del(&heap->heap_cdev);
+err1:
+ xa_erase(&eswin_heap_minors, minor);
+err0:
+ kfree(heap);
+ return err_ret;
+}
+
+int eswin_heap_delete(struct eswin_heap *heap)
+{
+ struct eswin_heap *h, *tmp;
+ int ret = -1;
+
+ if (!heap->name || !strcmp(heap->name, "")) {
+ pr_err("eswin_heap: Cannot delet heap without a name\n");
+ return -EINVAL;
+ }
+
+ /* find the heaplist by the heap name */
+ mutex_lock(&heap_list_lock);
+ list_for_each_entry_safe(h, tmp, &heap_list, list) {
+ if (!strcmp(h->name, heap->name)) {
+ pr_info("eswin_heap: deleted heap %s\n",
+ heap->name);
+ device_destroy(eswin_heap_class, h->heap_devt);
+ cdev_del(&h->heap_cdev);
+ xa_erase(&eswin_heap_minors, MINOR(h->heap_devt));
+ list_del(&h->list);
+ kfree(h);
+ ret = 0;
+ }
+ }
+ mutex_unlock(&heap_list_lock);
+
+ if (ret) {
+ pr_err("eswin_heap: heap named %s NOT found!\n", heap->name);
+ }
+
+ return ret;
+
+}
+
+int eswin_heap_delete_by_name(const char *name)
+{
+ struct eswin_heap *h, *tmp;
+ int ret = -1;
+
+ if (!name || !strcmp(name, "")) {
+ pr_err("eswin_heap: Cannot delet heap without a name\n");
+ return -EINVAL;
+ }
+
+ /* find the heaplist by the heap name */
+ mutex_lock(&heap_list_lock);
+ list_for_each_entry_safe(h, tmp, &heap_list, list) {
+ if (!strcmp(h->name, name)) {
+ pr_info("eswin_heap: deleted heap %s\n",
+ name);
+ device_destroy(eswin_heap_class, h->heap_devt);
+ cdev_del(&h->heap_cdev);
+ xa_erase(&eswin_heap_minors, MINOR(h->heap_devt));
+ list_del(&h->list);
+ kfree(h);
+ ret = 0;
+ }
+ }
+ mutex_unlock(&heap_list_lock);
+
+ if (ret) {
+ pr_err("eswin_heap: heap named %s NOT found!\n", name);
+ }
+
+ return ret;
+}
+
+int eswin_heap_kalloc(char *name, size_t len, unsigned int fd_flags, unsigned int heap_flags)
+{
+ struct eswin_heap *heap = NULL;
+#if 0
+ struct eswin_heap *h = NULL;
+ /* check the name is unique */
+ mutex_lock(&heap_list_lock);
+ list_for_each_entry(h, &heap_list, list) {
+ if (!strcmp(h->name, name)) {
+ heap = h;
+ break;
+ }
+ }
+ mutex_unlock(&heap_list_lock);
+#else
+ char *dev_path = NULL;
+ struct file *file;
+ int ret;
+
+ dev_path = kasprintf(GFP_KERNEL, "/dev/dma_heap/%s", name);
+ file = filp_open(dev_path, O_RDWR, 0);
+ if (IS_ERR(file)) {
+ ret = PTR_ERR(file);
+ pr_err("failed to open file %s: (%d)\n",
+ dev_path, ret);
+ return ret;
+ }
+ heap = file->private_data;
+#endif
+ if (!heap) {
+ printk("ERROR: Can't find this heap %s\n", name);
+ return -ENODEV;
+ }
+
+ return eswin_heap_buffer_alloc(heap, len, fd_flags, heap_flags);
+}
+EXPORT_SYMBOL(eswin_heap_kalloc);
+
+#if LINUX_VERSION_CODE >= KERNEL_VERSION(6,6,0)
+static char *eswin_heap_devnode(const struct device *dev, umode_t *mode)
+#else
+static char *eswin_heap_devnode(struct device *dev, umode_t *mode)
+#endif
+
+{
+ // return kasprintf(GFP_KERNEL, "eswin_heap/%s", dev_name(dev));
+ /* create device node under dma_heap instead of eswin_heap, so that memory lib can
+ avoid the diverseness.
+ */
+ return kasprintf(GFP_KERNEL, "dma_heap/%s", dev_name(dev));
+}
+
+int eswin_heap_init(void)
+{
+ int ret;
+
+ ret = alloc_chrdev_region(&eswin_heap_devt, 0, NUM_HEAP_MINORS, DEVNAME);
+ if (ret)
+ return ret;
+ #if LINUX_VERSION_CODE >= KERNEL_VERSION(6,6,0)
+ eswin_heap_class = class_create(DEVNAME);
+ #else
+ eswin_heap_class = class_create(THIS_MODULE, DEVNAME);
+ #endif
+ if (IS_ERR(eswin_heap_class)) {
+ unregister_chrdev_region(eswin_heap_devt, NUM_HEAP_MINORS);
+ return PTR_ERR(eswin_heap_class);
+ }
+ eswin_heap_class->devnode = eswin_heap_devnode;
+
+ return 0;
+}
+
+void eswin_heap_uninit(void)
+{
+ class_destroy(eswin_heap_class);
+}
diff --git a/drivers/memory/eswin/es_rsvmem_heap/eswin_rsvmem_heap.c b/drivers/memory/eswin/es_rsvmem_heap/eswin_rsvmem_heap.c
new file mode 100644
index 000000000000..804258f5ec74
--- /dev/null
+++ b/drivers/memory/eswin/es_rsvmem_heap/eswin_rsvmem_heap.c
@@ -0,0 +1,634 @@
+// SPDX-License-Identifier: GPL-2.0-only
+/*
+ * ESWIN rerserved memory heap.
+ * eswin_rsvmem_heap creates heap for the reserved memory that has compatible = "eswin-reserve-memory";
+ * property and no-map property.
+ *
+ * Copyright 2024 Beijing ESWIN Computing Technology Co., Ltd.
+ * Authors:
+ * LinMin<linmin@eswincomputing.com>
+ *
+ */
+
+#include <linux/dma-buf.h>
+#include <linux/dma-mapping.h>
+#include <linux/err.h>
+#include <linux/highmem.h>
+#include <linux/mm.h>
+#include <linux/module.h>
+#include <linux/scatterlist.h>
+#include <linux/slab.h>
+#include <linux/vmalloc.h>
+#include <linux/dma-map-ops.h>
+#include <linux/eswin_rsvmem_common.h>
+#include "../eswin_memblock.h"
+#include "../es_buddy/es_buddy.h"
+#include "include/uapi/linux/eswin_rsvmem_common.h"
+
+static const unsigned int orders[] = {MAX_ORDER-1, 9, 0};
+#define NUM_ORDERS ARRAY_SIZE(orders)
+
+struct eswin_rsvmem_heap {
+ struct eswin_heap *heap;
+ struct mem_block *memblock;
+};
+
+struct eswin_rsvmem_heap_buffer {
+ struct eswin_rsvmem_heap *heap;
+ struct list_head attachments;
+ struct mutex lock;
+ unsigned long len;
+ struct sg_table sg_table; // for buddy allocator
+ struct page **pages;
+ int vmap_cnt;
+ void *vaddr;
+ unsigned long fd_flags; // for vmap to determin the cache or non-cached mapping
+};
+
+struct eswin_heap_attachment {
+ struct device *dev;
+ struct sg_table *table;
+ struct list_head list;
+ bool mapped;
+};
+
+static struct sg_table *dup_sg_table(struct sg_table *table)
+{
+ struct sg_table *new_table;
+ int ret, i;
+ struct scatterlist *sg, *new_sg;
+
+ new_table = kzalloc(sizeof(*new_table), GFP_KERNEL);
+ if (!new_table)
+ return ERR_PTR(-ENOMEM);
+
+ ret = sg_alloc_table(new_table, table->orig_nents, GFP_KERNEL);
+ if (ret) {
+ kfree(new_table);
+ return ERR_PTR(-ENOMEM);
+ }
+
+ new_sg = new_table->sgl;
+ for_each_sgtable_sg(table, sg, i) {
+ sg_set_page(new_sg, sg_page(sg), sg->length, sg->offset);
+ new_sg = sg_next(new_sg);
+ }
+
+ return new_table;
+}
+
+static int eswin_rsvmem_heap_attach(struct dma_buf *dmabuf,
+ struct dma_buf_attachment *attachment)
+{
+ struct eswin_rsvmem_heap_buffer *buffer = dmabuf->priv;
+ struct eswin_heap_attachment *a;
+ struct sg_table *table;
+
+ a = kzalloc(sizeof(*a), GFP_KERNEL);
+ if (!a)
+ return -ENOMEM;
+
+ table = dup_sg_table(&buffer->sg_table);
+ if (IS_ERR(table)) {
+ kfree(a);
+ return -ENOMEM;
+ }
+
+ a->table = table;
+ a->dev = attachment->dev;
+ INIT_LIST_HEAD(&a->list);
+ a->mapped = false;
+
+ attachment->priv = a;
+
+ mutex_lock(&buffer->lock);
+ list_add(&a->list, &buffer->attachments);
+ mutex_unlock(&buffer->lock);
+
+ return 0;
+}
+
+static void eswin_rsvmem_heap_detach(struct dma_buf *dmabuf,
+ struct dma_buf_attachment *attachment)
+{
+ struct eswin_rsvmem_heap_buffer *buffer = dmabuf->priv;
+ struct eswin_heap_attachment *a = attachment->priv;
+
+ mutex_lock(&buffer->lock);
+ list_del(&a->list);
+ mutex_unlock(&buffer->lock);
+
+ sg_free_table(a->table);
+ kfree(a->table);
+ kfree(a);
+}
+
+static struct sg_table *eswin_rsvmem_heap_map_dma_buf(struct dma_buf_attachment *attachment,
+ enum dma_data_direction direction)
+{
+ struct eswin_heap_attachment *a = attachment->priv;
+ struct sg_table *table =a->table;
+ int ret;
+ unsigned long attrs = DMA_ATTR_SKIP_CPU_SYNC;
+
+ /* Skipt cache sync, since it takes a lot of time when import to device.
+ * It's the user's responsibility for guaranteeing the cache coherency by
+ flusing cache explicitly before importing to device.
+ */
+ ret = dma_map_sgtable(attachment->dev, table, direction, attrs);
+
+ if (ret)
+ return ERR_PTR(-ENOMEM);
+ a->mapped = true;
+ return table;
+}
+
+static void eswin_rsvmem_heap_unmap_dma_buf(struct dma_buf_attachment *attachment,
+ struct sg_table *table,
+ enum dma_data_direction direction)
+{
+ struct eswin_heap_attachment *a = attachment->priv;
+ unsigned long attrs = DMA_ATTR_SKIP_CPU_SYNC;
+
+ a->mapped = false;
+
+ /* Skipt cache sync, since it takes a lot of time when unmap from device.
+ * It's the user's responsibility for guaranteeing the cache coherency after
+ the device has done processing the data.(For example, CPU do NOT read untill
+ the device has done)
+ */
+ dma_unmap_sgtable(attachment->dev, table, direction, attrs);
+}
+
+static int eswin_rsvmem_dma_buf_begin_cpu_access(struct dma_buf *dmabuf,
+ enum dma_data_direction direction)
+{
+ struct eswin_rsvmem_heap_buffer *buffer = dmabuf->priv;
+ struct sg_table *table = &buffer->sg_table;
+ struct scatterlist *sg;
+ int i;
+
+ mutex_lock(&buffer->lock);
+
+ if (buffer->vmap_cnt)
+ invalidate_kernel_vmap_range(buffer->vaddr, buffer->len);
+
+ /* Since the cache sync was skipped when eswin_rsvmem_heap_map_dma_buf/eswin_rsvmem_heap_unmap_dma_buf,
+ So force cache sync here when user call ES_SYS_MemFlushCache, eventhough there
+ is no device attached to this dmabuf.
+ */
+ #ifndef QEMU_DEBUG
+ for_each_sg(table->sgl, sg, table->orig_nents, i)
+ arch_sync_dma_for_cpu(sg_phys(sg), sg->length, direction);
+
+ #endif
+ mutex_unlock(&buffer->lock);
+
+ return 0;
+}
+
+static int eswin_rsvmem_dma_buf_end_cpu_access(struct dma_buf *dmabuf,
+ enum dma_data_direction direction)
+{
+ struct eswin_rsvmem_heap_buffer *buffer = dmabuf->priv;
+ struct sg_table *table = &buffer->sg_table;
+ struct scatterlist *sg;
+ int i;
+
+ mutex_lock(&buffer->lock);
+
+ if (buffer->vmap_cnt)
+ flush_kernel_vmap_range(buffer->vaddr, buffer->len);
+
+ /* Since the cache sync was skipped while eswin_rsvmem_heap_map_dma_buf/eswin_rsvmem_heap_unmap_dma_buf,
+ So force cache sync here when user call ES_SYS_MemFlushCache, eventhough there
+ is no device attached to this dmabuf.
+ */
+ #ifndef QEMU_DEBUG
+ for_each_sg(table->sgl, sg, table->orig_nents, i)
+ arch_sync_dma_for_device(sg_phys(sg), sg->length, direction);
+ #endif
+ mutex_unlock(&buffer->lock);
+
+ return 0;
+}
+
+#if 0
+static int eswin_rsvmem_sync_cache_internal(struct dma_buf *dmabuf, enum dma_data_direction direction)
+{
+ struct eswin_rsvmem_heap_buffer *buffer = dmabuf->priv;
+ struct sg_table *table = &buffer->sg_table;
+ struct scatterlist *sg;
+ int i;
+
+ for_each_sg(table->sgl, sg, table->orig_nents, i)
+ arch_sync_dma_for_device(sg_phys(sg), sg->length, direction);
+
+
+ return 0;
+}
+#endif
+
+static int eswin_rsvmem_heap_mmap(struct dma_buf *dmabuf, struct vm_area_struct *vma)
+{
+ struct eswin_rsvmem_heap_buffer *buffer = dmabuf->priv;
+ struct eswin_heap *heap = buffer->heap->heap;
+ struct sg_table *table = &buffer->sg_table;
+ unsigned long addr = vma->vm_start;
+ unsigned long pgoff = vma->vm_pgoff, mapsize = 0;
+ unsigned long size_remaining = vma->vm_end - vma->vm_start;//vma_pages(vma);
+ struct scatterlist *sg;
+ struct page *page = NULL;
+ unsigned int nents = 0;
+ int i;
+ int ret;
+ const char *heap_name = NULL;
+
+ /* Mapping secure_memory with cached proprty to user space for CPU is NOT permitted */
+ heap_name = eswin_heap_get_name(heap);
+ if (unlikely(!strncmp("secure_memory", heap_name, 13))) {
+ if (!(vma->vm_flags & VM_NORESERVE))
+ return -EPERM;
+ }
+
+ if ((vma->vm_flags & (VM_SHARED | VM_MAYSHARE)) == 0)
+ return -EINVAL;
+
+ /* vm_private_data will be used by eswin-ipc-scpu.c.
+ ipc will import this dmabuf to get iova.
+ */
+ vma->vm_private_data = dmabuf;
+
+ /* support mman flag MAP_SHARED_VALIDATE | VM_NORESERVE, used to map uncached memory to user space.
+ Users should guarantee this buffer has been flushed to cache already.
+ */
+ if (vma->vm_flags & VM_NORESERVE) {
+ vm_flags_clear(vma, VM_NORESERVE);
+ #ifndef QEMU_DEBUG
+ vma->vm_page_prot = pgprot_dmacoherent(vma->vm_page_prot);
+ #endif
+ /* skip sync cache, users should guarantee the cache is clean after done using it in
+ cached mode(i.e, ES_SYS_Mmap(SYS_CACHE_MODE_CACHED))
+ */
+ }
+ pr_debug("%s, size_remaining:0x%lx, pgoff:0x%lx, dmabuf->size:0x%lx, start_phys:0x%llx\n",
+ __func__, size_remaining, pgoff, dmabuf->size, sg_phys(table->sgl));
+ for_each_sg(table->sgl, sg, table->orig_nents, i) {
+ pr_debug("sgl:%d, phys:0x%llx\n", i, sg_phys(sg));
+ if (pgoff >= (sg->length >> PAGE_SHIFT)) {
+ pgoff -= (sg->length >> PAGE_SHIFT);
+ continue;
+ }
+
+ page = sg_page(sg);
+ if (nents == 0) {
+ mapsize = sg->length - (pgoff << PAGE_SHIFT);
+ mapsize = min(size_remaining, mapsize);
+ ret = remap_pfn_range(vma, addr, page_to_pfn(page) + pgoff, mapsize,
+ vma->vm_page_prot);
+ pr_debug("nents:%d, sgl:%d, pgoff:0x%lx, mapsize:0x%lx, phys:0x%llx\n",
+ nents, i, pgoff, mapsize, pfn_to_phys(page_to_pfn(page) + pgoff));
+ }
+ else {
+ mapsize = min((unsigned int)size_remaining, (sg->length));
+ ret = remap_pfn_range(vma, addr, page_to_pfn(page), mapsize,
+ vma->vm_page_prot);
+ pr_debug("nents:%d, sgl:%d, mapsize:0x%lx, phys:0x%llx\n", nents, i, mapsize, page_to_phys(page));
+ }
+ pgoff = 0;
+ nents++;
+
+ if (ret)
+ return ret;
+
+ addr += mapsize;
+ size_remaining -= mapsize;
+ if (size_remaining == 0)
+ return 0;
+ }
+
+ return 0;
+}
+
+static void *eswin_rsvmem_heap_do_vmap(struct dma_buf *dmabuf)
+{
+ struct eswin_rsvmem_heap_buffer *buffer = dmabuf->priv;
+ pgprot_t prot = PAGE_KERNEL;
+ struct sg_table *table = &buffer->sg_table;
+ int npages = PAGE_ALIGN(buffer->len) / PAGE_SIZE;
+ struct page **pages = vmalloc(sizeof(struct page *) * npages);
+ struct page **tmp = pages;
+ struct sg_page_iter piter;
+ void *vaddr;
+
+ if (!pages)
+ return ERR_PTR(-ENOMEM);
+
+ for_each_sgtable_page(table, &piter, 0) {
+ WARN_ON(tmp - pages >= npages);
+ *tmp++ = sg_page_iter_page(&piter);
+ }
+
+ /* The property of this dmabuf in kernel space is determined by heap alloc with fd_flag. */
+ if (buffer->fd_flags & O_DSYNC) {
+ #ifndef QEMU_DEBUG
+ prot = pgprot_dmacoherent(PAGE_KERNEL);
+ #endif
+ pr_debug("%s syport uncached kernel dmabuf!, prot=0x%x\n", __func__, (unsigned int)pgprot_val(prot));
+ }
+ else {
+ pr_debug("%s memport cached kernel dmabuf!\n", __func__);
+ }
+
+ vaddr = vmap(pages, npages, VM_MAP, prot);
+ vfree(pages);
+
+ if (!vaddr)
+ return ERR_PTR(-ENOMEM);
+
+ return vaddr;
+}
+
+static int eswin_rsvmem_heap_vmap(struct dma_buf *dmabuf, struct dma_buf_map *map)
+{
+ struct eswin_rsvmem_heap_buffer *buffer = dmabuf->priv;
+ void *vaddr;
+ int ret = 0;
+
+ mutex_lock(&buffer->lock);
+ if (buffer->vmap_cnt) {
+ buffer->vmap_cnt++;
+ dma_buf_map_set_vaddr(map, buffer->vaddr);
+ goto out;
+ }
+
+ vaddr = eswin_rsvmem_heap_do_vmap(dmabuf);
+ if (IS_ERR(vaddr)) {
+ ret = PTR_ERR(vaddr);
+ goto out;
+ }
+ buffer->vaddr = vaddr;
+ buffer->vmap_cnt++;
+ dma_buf_map_set_vaddr(map, buffer->vaddr);
+out:
+ mutex_unlock(&buffer->lock);
+
+ return ret;
+}
+
+static void eswin_rsvmem_heap_vunmap(struct dma_buf *dmabuf, struct dma_buf_map *map)
+{
+ struct eswin_rsvmem_heap_buffer *buffer = dmabuf->priv;
+
+ mutex_lock(&buffer->lock);
+ if (!--buffer->vmap_cnt) {
+ vunmap(buffer->vaddr);
+ buffer->vaddr = NULL;
+ }
+ mutex_unlock(&buffer->lock);
+ dma_buf_map_clear(map);
+}
+
+static void eswin_rsvmem_heap_dma_buf_release(struct dma_buf *dmabuf)
+{
+ struct eswin_rsvmem_heap_buffer *buffer = dmabuf->priv;
+ struct sg_table *table;
+ struct scatterlist *sg;
+ int i;
+
+ table = &buffer->sg_table;
+ if (buffer->vmap_cnt > 0) {
+ WARN(1, "%s: buffer still mapped in the kernel\n", __func__);
+ vunmap(buffer->vaddr);
+ buffer->vaddr = NULL;
+ }
+
+ for_each_sgtable_sg(table, sg, i) {
+ struct page *page = sg_page(sg);
+ // pr_debug("%s:%d,page_size(page)=0x%lx, phys_addr=0x%llx\n",
+ // __func__, __LINE__, page_size(page), page_to_phys(page));
+ es_free_pages(buffer->heap->memblock, page);
+ }
+ sg_free_table(table);
+
+ kfree(buffer);
+}
+
+static const struct dma_buf_ops eswin_rsvmem_heap_buf_ops = {
+ .attach = eswin_rsvmem_heap_attach,
+ .detach = eswin_rsvmem_heap_detach,
+ .map_dma_buf = eswin_rsvmem_heap_map_dma_buf,
+ .unmap_dma_buf = eswin_rsvmem_heap_unmap_dma_buf,
+ .begin_cpu_access = eswin_rsvmem_dma_buf_begin_cpu_access,
+ .end_cpu_access = eswin_rsvmem_dma_buf_end_cpu_access,
+ .mmap = eswin_rsvmem_heap_mmap,
+ .vmap = eswin_rsvmem_heap_vmap,
+ .vunmap = eswin_rsvmem_heap_vunmap,
+ .release = eswin_rsvmem_heap_dma_buf_release,
+};
+
+static struct page *alloc_largest_available(struct mem_block *memblock,
+ unsigned long size,
+ unsigned int max_order)
+{
+ struct page *page;
+ int i;
+
+ for (i = 0; i < NUM_ORDERS; i++) {
+ if (size < (PAGE_SIZE << orders[i]))
+ continue;
+ if (max_order < orders[i])
+ continue;
+
+ page = es_alloc_pages(memblock, orders[i]);
+ if (!page)
+ continue;
+ return page;
+ }
+ return NULL;
+}
+
+static struct dma_buf *eswin_rsvmem_heap_allocate(struct eswin_heap *heap,
+ unsigned long len,
+ unsigned long fd_flags,
+ unsigned long heap_flags)
+{
+ struct eswin_rsvmem_heap *rsvmem_heap = eswin_heap_get_drvdata(heap);
+ struct eswin_rsvmem_heap_buffer *buffer;
+ DEFINE_DMA_BUF_EXPORT_INFO(exp_info);
+ unsigned long size_remaining = len;
+ unsigned int max_order = orders[0];
+ struct dma_buf *dmabuf;
+ struct sg_table *table;
+ struct scatterlist *sg;
+ struct list_head pages;
+ struct page *page, *tmp_page;
+ int i, ret = -ENOMEM;
+ const char *heap_name = NULL;
+
+ /* Mapping secure_memory with cached proprty to kernel space for CPU is NOT permitted */
+ heap_name = eswin_heap_get_name(rsvmem_heap->heap);
+ if (unlikely(!strncmp("secure_memory", heap_name, 13))) {
+ if (!(fd_flags & O_DSYNC))
+ return ERR_PTR(-EINVAL);
+ }
+
+ buffer = kzalloc(sizeof(*buffer), GFP_KERNEL);
+ if (!buffer)
+ return ERR_PTR(-ENOMEM);
+
+ INIT_LIST_HEAD(&buffer->attachments);
+ mutex_init(&buffer->lock);
+ buffer->heap = rsvmem_heap;
+ buffer->len = len;
+ buffer->fd_flags = fd_flags;
+
+ INIT_LIST_HEAD(&pages);
+ i = 0;
+ while (size_remaining > 0) {
+ /*
+ * Avoid trying to allocate memory if the process
+ * has been killed by SIGKILL
+ */
+ if (fatal_signal_pending(current)) {
+ ret = -EINTR;
+ goto free_buffer;
+ }
+
+ page = alloc_largest_available(rsvmem_heap->memblock, size_remaining, max_order);
+ if (!page)
+ goto free_buffer;
+
+ list_add_tail(&page->lru, &pages);
+ size_remaining -= page_size(page);
+ max_order = compound_order(page);
+ i++;
+ }
+
+ table = &buffer->sg_table;
+ if (sg_alloc_table(table, i, GFP_KERNEL))
+ goto free_buffer;
+
+ sg = table->sgl;
+ list_for_each_entry_safe(page, tmp_page, &pages, lru) {
+ sg_set_page(sg, page, page_size(page), 0);
+ sg = sg_next(sg);
+ list_del(&page->lru);
+ }
+
+ /* create the dmabuf */
+ exp_info.exp_name = eswin_heap_get_name(heap);
+ exp_info.ops = &eswin_rsvmem_heap_buf_ops;
+ exp_info.size = buffer->len;
+ exp_info.flags = O_RDWR | O_CLOEXEC;
+ exp_info.priv = buffer;
+ dmabuf = dma_buf_export(&exp_info);
+ if (IS_ERR(dmabuf)) {
+ ret = PTR_ERR(dmabuf);
+ goto free_pages;
+ }
+ return dmabuf;
+
+free_pages:
+ for_each_sgtable_sg(table, sg, i) {
+ struct page *p = sg_page(sg);
+
+ es_free_pages(rsvmem_heap->memblock, p);
+ }
+ sg_free_table(table);
+free_buffer:
+ list_for_each_entry_safe(page, tmp_page, &pages, lru)
+ es_free_pages(rsvmem_heap->memblock, page);
+ kfree(buffer);
+
+ return ERR_PTR(ret);
+}
+
+static const struct eswin_heap_ops eswin_rsvmem_heap_ops = {
+ .allocate = eswin_rsvmem_heap_allocate,
+};
+
+static int __add_eswin_rsvmem_heap(struct mem_block *memblock, void *data)
+{
+ struct eswin_rsvmem_heap *rsvmem_heap;
+ struct eswin_heap_export_info exp_info;
+
+ rsvmem_heap = kzalloc(sizeof(*rsvmem_heap), GFP_KERNEL);
+ if (!rsvmem_heap)
+ return -ENOMEM;
+ rsvmem_heap->memblock = memblock;
+
+ exp_info.name = eswin_rsvmem_get_name(memblock);
+ exp_info.ops = &eswin_rsvmem_heap_ops;
+ exp_info.priv = rsvmem_heap;
+
+ rsvmem_heap->heap = eswin_heap_add(&exp_info);
+ if (IS_ERR(rsvmem_heap->heap)) {
+ int ret = PTR_ERR(rsvmem_heap->heap);
+
+ kfree(rsvmem_heap);
+ return ret;
+ }
+
+ pr_info("%s for %s successfully!\n", __func__, exp_info.name);
+
+ return 0;
+}
+
+static char *es_heap_name_prefix[] = {
+ "mmz_nid_",
+ "secure_memory"
+};
+#define NUM_ESWIN_RSVMEM_HEAPS ARRAY_SIZE(es_heap_name_prefix)
+
+static int do_add_eswin_rsvmem_heap(struct mem_block *memblock, void *data)
+{
+ int ret = 0;
+ char *prefix = data;
+ const char *rsvmem_name = eswin_rsvmem_get_name(memblock);
+
+ if (strncmp(rsvmem_name, prefix, strlen(prefix)) == 0)
+ ret = __add_eswin_rsvmem_heap(memblock, NULL);
+
+ return ret;
+}
+static int add_eswin_rsvmem_heaps(void)
+{
+ int i;
+ int ret;
+
+ ret = eswin_heap_init();
+ if (ret)
+ return ret;
+
+ for (i = 0; i < NUM_ESWIN_RSVMEM_HEAPS; i++) {
+ eswin_rsvmem_for_each_block(do_add_eswin_rsvmem_heap, es_heap_name_prefix[i]);
+ }
+
+ return 0;
+}
+
+static int do_delete_eswin_rsvmem_heap(struct mem_block *memblock, void *data)
+{
+ int ret = 0;
+ char *prefix = data;
+ const char *rsvmem_name = eswin_rsvmem_get_name(memblock);
+
+ if (strncmp(rsvmem_name, prefix, strlen(prefix)) == 0)
+ ret = eswin_heap_delete_by_name(rsvmem_name);
+
+ return ret;
+}
+static void __exit delete_eswin_rsvmem_heaps(void)
+{
+ int i;
+
+ for (i = 0; i < NUM_ESWIN_RSVMEM_HEAPS; i++) {
+ eswin_rsvmem_for_each_block(do_delete_eswin_rsvmem_heap, es_heap_name_prefix[i]);
+ }
+ eswin_heap_uninit();
+}
+module_init(add_eswin_rsvmem_heaps);
+module_exit(delete_eswin_rsvmem_heaps);
+MODULE_IMPORT_NS(DMA_BUF);
+MODULE_LICENSE("GPL v2");
diff --git a/drivers/memory/eswin/es_rsvmem_heap/include/linux/mem_perf_api.h b/drivers/memory/eswin/es_rsvmem_heap/include/linux/mem_perf_api.h
new file mode 100644
index 000000000000..a0357b43d914
--- /dev/null
+++ b/drivers/memory/eswin/es_rsvmem_heap/include/linux/mem_perf_api.h
@@ -0,0 +1,123 @@
+// SPDX-License-Identifier: GPL-2.0-only
+/*
+ * Performance test APIs for ESWIN memory
+ *
+ * Copyright 2024 Beijing ESWIN Computing Technology Co., Ltd.
+ * Authors:
+ * LinMin<linmin@eswincomputing.com>
+ *
+ */
+#ifndef __MEM_PERF_API_H__
+#define __MEM_PERF_API_H__
+
+#define IN_KERNEL 1
+
+#if IN_KERNEL
+#include <linux/list.h>
+#define alloc_mem_perf_info(size) kmalloc(size, GFP_KERNEL)
+#define free_mem_perf_info(info) kfree(info)
+#define PRINT_INFO(fmt, ...) pr_info(fmt, ##__VA_ARGS__)
+#else
+#include <stdlib.h>
+#include <stdio.h>
+#include <string.h>
+#include "list.h"
+#define alloc_mem_perf_info(size) malloc(size)
+#define free_mem_perf_info(info) free(info)
+#define PRINT_INFO(fmt, ...) printf("%s" fmt, "[ES_DMA_INF]", ##__VA_ARGS__)
+#endif
+
+typedef unsigned long long uint64;
+
+struct mem_perf_info {
+ struct list_head node;
+ char func_name[64];
+ uint64 cycles_start;
+ uint64 cycles_end;
+ uint64 cycles_elapased;
+};
+
+#if defined(CONFIG_RISCV)
+static inline int metal_timer_get_cyclecount(unsigned long long *mcc)
+{
+ unsigned long cycles;
+ asm volatile ("rdtime %0" : "=r" (cycles));
+ *mcc = cycles;
+
+ return 0;
+}
+#else
+static inline int metal_timer_get_cyclecount(unsigned long long *mcc)
+{
+ return 0;
+}
+#endif
+
+//struct mem_perf_info *lookup_mem_api_from_list()
+static inline struct mem_perf_info *memperf_record_cycle_start(const char *func_name, struct list_head *mem_perf_list_head)
+{
+ struct mem_perf_info *m_perf_i;
+
+ m_perf_i = alloc_mem_perf_info(sizeof(*m_perf_i));
+ if (!m_perf_i) {
+ PRINT_INFO("mem perf info alloc failed!\n");
+ return NULL;
+ }
+
+ sprintf(m_perf_i->func_name, "%s", func_name);
+ list_add_tail(&m_perf_i->node, mem_perf_list_head);
+ metal_timer_get_cyclecount(&m_perf_i->cycles_start);
+ m_perf_i->cycles_end = m_perf_i->cycles_start;
+
+ return m_perf_i;
+}
+
+static inline int memperf_record_cycle_end(struct mem_perf_info *m_perf_i)
+{
+ if (NULL == m_perf_i)
+ return -1;
+
+ metal_timer_get_cyclecount(&m_perf_i->cycles_end);
+
+ return 0;
+}
+
+#if defined(CONFIG_RISCV)
+static inline int memperf_print_records(struct list_head *mem_perf_list_head)
+{
+ struct mem_perf_info *m_perf_i;
+ uint64 total_cycles = 0;
+
+ list_for_each_entry(m_perf_i, mem_perf_list_head, node) {
+ m_perf_i->cycles_elapased = m_perf_i->cycles_end - m_perf_i->cycles_start;
+ total_cycles += m_perf_i->cycles_elapased;
+ }
+ PRINT_INFO("Total cycles:%lld, %lld us\n", total_cycles, total_cycles*1000*1000/32768);
+ list_for_each_entry(m_perf_i, mem_perf_list_head, node) {
+ PRINT_INFO("cycle_elapsed:%lld---%%\%lld.%2lld, %s\n",
+ m_perf_i->cycles_elapased, (100*m_perf_i->cycles_elapased)/total_cycles,
+ (10000*m_perf_i->cycles_elapased)%total_cycles, m_perf_i->func_name);
+ }
+
+ return 0;
+}
+#else
+static inline int memperf_print_records(struct list_head *mem_perf_list_head)
+{
+ return 0;
+}
+#endif
+
+static inline int memperf_free_records_list(struct list_head *mem_perf_list_head)
+{
+ struct mem_perf_info *m_perf_i, *tmp;
+
+ list_for_each_entry_safe(m_perf_i, tmp, mem_perf_list_head, node) {
+ list_del(&m_perf_i->node);
+ free_mem_perf_info(m_perf_i);
+ }
+
+ return 0;
+}
+
+#endif
diff --git a/drivers/memory/eswin/es_rsvmem_heap/include/uapi/linux/eswin_rsvmem_common.h b/drivers/memory/eswin/es_rsvmem_heap/include/uapi/linux/eswin_rsvmem_common.h
new file mode 100644
index 000000000000..7d863001cfc3
--- /dev/null
+++ b/drivers/memory/eswin/es_rsvmem_heap/include/uapi/linux/eswin_rsvmem_common.h
@@ -0,0 +1,56 @@
+// SPDX-License-Identifier: GPL-2.0-only
+/*
+ * ESWIN Heaps Userspace API
+ *
+ * Copyright 2024 Beijing ESWIN Computing Technology Co., Ltd.
+ * Authors:
+ * LinMin<linmin@eswincomputing.com>
+ *
+ */
+
+#ifndef _UAPI_ESWIN_HEAPS_H
+#define _UAPI_ESWIN_HEAPS_H
+
+#include <linux/ioctl.h>
+#include <linux/types.h>
+
+/**
+ * DOC: DMABUF Heaps Userspace API
+ */
+/* Valid FD_FLAGS are O_CLOEXEC, O_RDONLY, O_WRONLY, O_RDWR, O_SYNC */
+#define ESWIN_HEAP_VALID_FD_FLAGS (O_CLOEXEC | O_ACCMODE | O_SYNC)
+
+/* Add HEAP_SPRAM_FORCE_CONTIGUOUS heap flags for ESWIN SPRAM HEAP */
+#define HEAP_FLAGS_SPRAM_FORCE_CONTIGUOUS (1 << 0)
+#define ESWIN_HEAP_VALID_HEAP_FLAGS (HEAP_FLAGS_SPRAM_FORCE_CONTIGUOUS)
+
+/**
+ * struct eswin_heap_allocation_data - metadata passed from userspace for
+ * allocations
+ * @len: size of the allocation
+ * @fd: will be populated with a fd which provides the
+ * handle to the allocated dma-buf
+ * @fd_flags: file descriptor flags used when allocating
+ * @heap_flags: flags passed to heap
+ *
+ * Provided by userspace as an argument to the ioctl
+ */
+struct eswin_heap_allocation_data {
+ __u64 len;
+ __u32 fd;
+ __u32 fd_flags;
+ __u64 heap_flags;
+};
+
+#define ESWIN_HEAP_IOC_MAGIC 'H'
+
+/**
+ * DOC: ESWIN_HEAP_IOCTL_ALLOC - allocate memory from pool
+ *
+ * Takes a eswin_heap_allocation_data struct and returns it with the fd field
+ * populated with the dmabuf handle of the allocation.
+ */
+#define ESWIN_HEAP_IOCTL_ALLOC _IOWR(ESWIN_HEAP_IOC_MAGIC, 0x0,\
+ struct eswin_heap_allocation_data)
+
+#endif /* _UAPI_ESWIN_HEAPS_H */
diff --git a/include/linux/dmabuf-heap-import-helper.h b/include/linux/dmabuf-heap-import-helper.h
new file mode 100644
index 000000000000..6fd339ee8ff3
--- /dev/null
+++ b/include/linux/dmabuf-heap-import-helper.h
@@ -0,0 +1,100 @@
+#ifndef _DMABUF_HEAP_IMPORT_H_
+#define _DMABUF_HEAP_IMPORT_H_
+
+#include <linux/rbtree.h>
+#include <linux/dma-buf.h>
+#include <linux/mutex.h>
+#include <linux/types.h>
+#include <linux/scatterlist.h>
+#include <linux/eswin_rsvmem_common.h>
+
+#define SYSTEM_DEV_NODE "system"
+#define CMA_DEV_NODE_RES "reserved"
+#define CMA_DEV_NODE_DFT "linux,cma"
+#define SYSTEM_COHERENT_DEV_NODE "system_coherent"
+
+struct dmaheap_file_private {
+ /* private: */
+ struct rb_root dmabufs;
+ struct rb_root handles;
+};
+
+struct heap_root {
+ struct dmaheap_file_private fp;
+ struct device *dev;
+ struct list_head header;
+ struct mutex lock;
+};
+
+struct heap_mem {
+ /* refcount is also protected by lock in the struct heap_root */
+ struct kref refcount;
+
+ int dbuf_fd;
+ struct dma_buf *dbuf;
+ struct dma_buf_attachment *import_attach;
+
+ struct heap_root *root;
+ struct list_head list;
+ struct rb_node *rb;
+
+ struct sg_table *sgt;
+ void *vaddr;
+
+ enum dma_data_direction dir;
+};
+
+struct esw_export_buffer_info {
+ char name[64];
+ int fd_flags;
+
+ int dbuf_fd;
+ struct dma_buf *dmabuf;
+
+ struct esw_slice_buffer {
+ __u64 offset;
+ size_t len;
+ } slice;
+};
+
+static int inline common_dmabuf_heap_begin_cpu_access(struct heap_mem *heap_obj)
+{
+ return dma_buf_begin_cpu_access(heap_obj->dbuf, heap_obj->dir);
+}
+
+static int inline common_dmabuf_heap_end_cpu_access(struct heap_mem *heap_obj)
+{
+ return dma_buf_end_cpu_access(heap_obj->dbuf, heap_obj->dir);
+}
+
+static inline size_t common_dmabuf_heap_get_size(struct heap_mem *heap_obj)
+{
+ WARN_ON(!heap_obj);
+ return (heap_obj != NULL) ? heap_obj->dbuf->size : 0;
+}
+
+static inline void common_dmabuf_heap_set_dir(struct heap_mem *heap_obj, enum dma_data_direction dir)
+{
+ WARN_ON(!heap_obj);
+ if (heap_obj)
+ heap_obj->dir = dir;
+}
+
+void common_dmabuf_heap_import_init(struct heap_root *root, struct device *dev);
+void common_dmabuf_heap_import_uninit(struct heap_root *root);
+
+struct heap_mem *common_dmabuf_lookup_heapobj_by_fd(struct heap_root *root, int fd);
+struct heap_mem *common_dmabuf_lookup_heapobj_by_dma_buf_st(struct heap_root *root, struct dma_buf *dma_buf);
+
+struct heap_mem *common_dmabuf_heap_import_from_user(struct heap_root *root, int fd);
+struct heap_mem *common_dmabuf_heap_import_from_user_with_dma_buf_st(struct heap_root *root, struct dma_buf *dma_buf);
+void common_dmabuf_heap_release(struct heap_mem *heap_obj);
+
+void *common_dmabuf_heap_map_vaddr(struct heap_mem *heap_obj);
+void common_dmabuf_heap_umap_vaddr(struct heap_mem *heap_obj);
+
+struct heap_mem *common_dmabuf_heap_import_from_kernel(struct heap_root *root, char *name, size_t len, unsigned int fd_flags);
+
+int esw_common_dmabuf_split_export(int dbuf_fd, unsigned int offset, size_t len, int fd_flags, char *name);
+
+#endif
diff --git a/include/linux/eswin_rsvmem_common.h b/include/linux/eswin_rsvmem_common.h
new file mode 100644
index 000000000000..33862d7292d7
--- /dev/null
+++ b/include/linux/eswin_rsvmem_common.h
@@ -0,0 +1,94 @@
+/* SPDX-License-Identifier: GPL-2.0 */
+/*
+ * DMABUF Heaps Allocation Infrastructure
+ *
+ * Copyright (C) 2011 Google, Inc.
+ * Copyright (C) 2019 Linaro Ltd.
+ */
+
+#ifndef _ESWIN_HEAPS_H
+#define _ESWIN_HEAPS_H
+
+#include <linux/cdev.h>
+#include <linux/types.h>
+#include <linux/version.h>
+
+#if LINUX_VERSION_CODE >= KERNEL_VERSION(6,6,0)
+#define dma_buf_map iosys_map
+#define dma_buf_map_set_vaddr iosys_map_set_vaddr
+#define dma_buf_map_clear iosys_map_clear
+#else
+#define vm_flags_clear(vma, flags) (vma->vm_flags &= ~flags)
+#endif
+
+struct eswin_heap;
+
+/**
+ * struct eswin_heap_ops - ops to operate on a given heap
+ * @allocate: allocate dmabuf and return struct dma_buf ptr
+ *
+ * allocate returns dmabuf on success, ERR_PTR(-errno) on error.
+ */
+struct eswin_heap_ops {
+ struct dma_buf *(*allocate)(struct eswin_heap *heap,
+ unsigned long len,
+ unsigned long fd_flags,
+ unsigned long heap_flags);
+};
+
+/**
+ * struct eswin_heap_export_info - information needed to export a new dmabuf heap
+ * @name: used for debugging/device-node name
+ * @ops: ops struct for this heap
+ * @priv: heap exporter private data
+ *
+ * Information needed to export a new dmabuf heap.
+ */
+struct eswin_heap_export_info {
+ const char *name;
+ const struct eswin_heap_ops *ops;
+ void *priv;
+};
+
+/**
+ * eswin_heap_get_drvdata() - get per-heap driver data
+ * @heap: DMA-Heap to retrieve private data for
+ *
+ * Returns:
+ * The per-heap data for the heap.
+ */
+void *eswin_heap_get_drvdata(struct eswin_heap *heap);
+
+/**
+ * eswin_heap_get_name() - get heap name
+ * @heap: DMA-Heap to retrieve private data for
+ *
+ * Returns:
+ * The char* for the heap name.
+ */
+const char *eswin_heap_get_name(struct eswin_heap *heap);
+
+/**
+ * eswin_heap_add - adds a heap to dmabuf heaps
+ * @exp_info: information needed to register this heap
+ */
+struct eswin_heap *eswin_heap_add(const struct eswin_heap_export_info *exp_info);
+
+/**
+ * eswin_heap_delete - delete a heap from dmabuf heaps
+ * @heap: heap needed to delete
+ */
+int eswin_heap_delete(struct eswin_heap *heap);
+
+/**
+ * eswin_heap_delete_by_name - find and delete a heap from dmabuf heaps by name
+ * @name: heap name needed to delete
+ */
+int eswin_heap_delete_by_name(const char *name);
+
+int eswin_heap_kalloc(char *name, size_t len, unsigned int fd_flags, unsigned int heap_flags);
+
+int eswin_heap_init(void);
+void eswin_heap_uninit(void);
+#define dma_heap_kalloc(name, len, fd_flags, heap_flags) eswin_heap_kalloc(name, len, fd_flags, heap_flags)
+#endif /* _ESWIN_HEAPS_H */
diff --git a/include/uapi/linux/es_vb_user.h b/include/uapi/linux/es_vb_user.h
new file mode 100644
index 000000000000..db3bda6d0d69
--- /dev/null
+++ b/include/uapi/linux/es_vb_user.h
@@ -0,0 +1,58 @@
+#ifndef ES_VB_USER_H
+#define ES_VB_USER_H
+
+/**
+ * mmz vb configurations
+*/
+#define ES_VB_MAX_MMZs 4
+
+#define ES_VB_INVALID_POOLID (-1U)
+
+#define ES_VB_MAX_MOD_POOL 16
+
+#define ES_MAX_MMZ_NAME_LEN 64
+
+/**
+ * mmz vb pool or block struct definition
+ */
+typedef enum esVB_UID_E {
+ VB_UID_PRIVATE = 0,
+ VB_UID_COMMON,
+ VB_UID_VI,
+ VB_UID_VO,
+ VB_UID_VPS,
+ VB_UID_VENC,
+ VB_UID_VDEC,
+ VB_UID_HAE,
+ VB_UID_USER,
+ VB_UID_BUTT,
+ VB_UID_MAX,
+} VB_UID_E;
+
+typedef enum {
+ SYS_CACHE_MODE_NOCACHE = 0,
+ SYS_CACHE_MODE_CACHED = 1,
+ SYS_CACHE_MODE_LLC = 2,
+ SYS_CACHE_MODE_BUTT,
+} SYS_CACHE_MODE_E;
+
+typedef struct esVB_POOL_CONFIG_S {
+ ES_U64 blkSize;
+ ES_U32 blkCnt;
+ SYS_CACHE_MODE_E enRemapMode;
+ ES_CHAR mmzName[ES_MAX_MMZ_NAME_LEN];
+} VB_POOL_CONFIG_S;
+
+typedef struct esVB_CONFIG_S {
+ ES_U32 poolCnt;
+ VB_POOL_CONFIG_S poolCfgs[ES_VB_MAX_MOD_POOL];
+} VB_CONFIG_S;
+
+typedef struct ES_DEV_BUF {
+ ES_U64 memFd;
+ ES_U64 offset;
+ ES_U64 size;
+ ES_U64 reserve;
+} ES_DEV_BUF_S;
+
+#endif
diff --git a/include/uapi/linux/mmz_vb.h b/include/uapi/linux/mmz_vb.h
new file mode 100644
index 000000000000..49ebac96f6e1
--- /dev/null
+++ b/include/uapi/linux/mmz_vb.h
@@ -0,0 +1,175 @@
+#ifndef _MMZ_VB_UAPI_H_
+#define _MMZ_VB_UAPI_H_
+
+#include <linux/types.h>
+
+/*vb cfg flag*/
+#define MMZ_VB_CFG_FLAG_INIT (1 << 0)
+
+/*vb pool flag*/
+#define MMZ_VB_POOL_FLAG_DESTORY (1 << 0)
+
+/*set cfg cmd*/
+typedef struct esVB_SET_CFG_REQ_S {
+ enum esVB_UID_E uid;
+ struct esVB_CONFIG_S cfg;
+}VB_SET_CFG_REQ_S;
+
+typedef struct esVB_SET_CFG_CMD_S {
+ struct esVB_SET_CFG_REQ_S CfgReq;
+}VB_SET_CFG_CMD_S;
+
+/*get cfg cmd*/
+typedef struct esVB_GET_CFG_REQ_S {
+ enum esVB_UID_E uid;
+}VB_Get_CFG_REQ_S;
+
+typedef struct esVB_GET_CFG_RSP_S {
+ struct esVB_CONFIG_S cfg;
+}VB_Get_CFG_RSP_S;
+
+typedef struct esVB_GET_CFG_CMD_S {
+ struct esVB_GET_CFG_REQ_S req;
+ struct esVB_GET_CFG_RSP_S rsp;
+}VB_GET_CFG_CMD_S;
+
+/*Init cfg cmd*/
+typedef struct esVB_INIT_CFG_REQ_S {
+ enum esVB_UID_E uid;
+}VB_INIT_CFG_REQ_S;
+
+typedef struct esVB_INIT_CFG_CMD_S {
+ struct esVB_INIT_CFG_REQ_S req;
+}VB_INIT_CFG_CMD_S;
+
+/*UnInit cfg cmd*/
+typedef struct esVB_UNINIT_CFG_REQ_S {
+ enum esVB_UID_E uid;
+}VB_UNINIT_CFG_REQ_S;
+
+typedef struct esVB_UNINIT_CFG_CMD_S {
+ struct esVB_UNINIT_CFG_REQ_S req;
+}VB_UNINIT_CFG_CMD_S;
+
+/*create pool cmd*/
+typedef struct esVB_CREATE_POOL_REQ_S {
+ struct esVB_POOL_CONFIG_S req;
+}VB_CREATE_POOL_REQ_S;
+
+typedef struct esVB_CREATE_POOL_RESP_S {
+ __u32 PoolId;
+} VB_CREATE_POOL_RESP_S;
+
+typedef struct esVB_CREATE_POOL_CMD_S {
+ struct esVB_CREATE_POOL_REQ_S PoolReq;
+ struct esVB_CREATE_POOL_RESP_S PoolResp;
+}VB_CREATE_POOL_CMD_S;
+
+/*destory pool cmd*/
+typedef struct esVB_DESTORY_POOL_REQ_S {
+ __u32 PoolId;
+}VB_DESTORY_POOL_REQ_S;
+
+typedef struct esVB_DESTORY_POOL_RESP_S {
+ __u32 Result;
+}VB_DESTORY_POOL_RESP_S;
+
+typedef struct esVB_DESTORY_POOL_CMD_S {
+ struct esVB_DESTORY_POOL_REQ_S req;
+ struct esVB_DESTORY_POOL_RESP_S rsp;
+}VB_DESTORY_POOL_CMD_S;
+
+typedef struct esVB_GET_BLOCK_REQ_S {
+ enum esVB_UID_E uid;
+ VB_POOL poolId;
+ __u64 blkSize;
+ char mmzName[ES_MAX_MMZ_NAME_LEN];
+}VB_GET_BLOCK_REQ_S;
+typedef struct esVB_GET_BLOCK_RESP_S {
+ __u64 actualBlkSize;
+ int fd;
+ int nr; /*bitmap index in the pool*/
+}VB_GET_BLOCK_RESP_S;
+typedef struct esVB_GET_BLOCK_CMD_S
+{
+ struct esVB_GET_BLOCK_REQ_S getBlkReq;
+ struct esVB_GET_BLOCK_RESP_S getBlkResp;
+}VB_GET_BLOCK_CMD_S;
+
+//corresponding to MMZ_VB_IOCTL_POOL_SIZE
+typedef struct esVB_GET_POOLSIZE_CMD_S
+{
+ VB_POOL poolId;
+ __u64 poolSize;
+}VB_GET_POOLSIZE_CMD_S;
+
+//corresponding to MMZ_VB_IOCTL_FLUSH_POOL
+typedef struct esVB_FLUSH_POOL_CMD_S
+{
+ VB_POOL poolId;
+ __u64 offset; // offset addr in the pool
+ __u64 size; // size to be flushed
+}VB_FLUSH_POOL_CMD_S;
+
+//corresponding to MMZ_VB_IOCTL_BLOCK_TO_POOL
+typedef struct esVB_BLOCK_TO_POOL_CMD_S
+{
+ int fd; // Input: The dmabuf_fd of the block
+ VB_POOL poolId; //Output: The pool which the block belongs to;
+}VB_BLOCK_TO_POOL_CMD_S;
+
+//corresponding to MMZ_VB_IOCTL_GET_BLOCK_OFFSET
+typedef struct esVB_GET_BLOCKOFFSET_CMD_S
+{
+ int fd; // Input: The dmabuf_fd, it might be the real block or the splittedBlock
+ __u64 offset; // Output: The offset in pool
+}VB_GET_BLOCKOFFSET_CMD_S;
+
+//corresponding to MMZ_VB_IOCTL_SPLIT_DMABUF
+typedef struct esVB_SPLIT_DMABUF_CMD_S {
+ int fd; /* Input: The original dmabuf fd to be splitted */
+ int slice_fd; /* Outpu: splitted dmabuf fd */
+ __u64 offset; /* Input: offset of the buffer relative to the original dmabuf */
+ __u64 len; /* size of the buffer to be splitted */
+}VB_BLOCK_SPLIT_CMD_S;
+
+//corresponding to MMZ_VB_IOCTL_DMABUF_REFCOUNT
+typedef struct esVB_DMABUF_REFCOUNT_CMD_S
+{
+ int fd; // Input: The dmabuf_fd
+ __u64 refCnt; // Output: The file_count of the dmabuf
+}VB_DMABUF_REFCOUNT_CMD_S;
+
+//corresponding to MMZ_VB_IOCTL_RETRIEVE_MEM_NODE
+typedef struct esVB_RETRIEVE_MEM_NODE_CMD_S
+{
+ int fd; // Input: The dmabuf_fd
+ void *cpu_vaddr; // Input: The virtual addr of cpu in user space
+ int numa_node; // Ouput: return the NUMA node id of the memory
+}VB_RETRIEVE_MEM_NODE_CMD_S;
+
+//corresponding to MMZ_VB_IOCTL_DMABUF_SIZE
+typedef struct esVB_DMABUF_SIZE_CMD_S
+{
+ int fd; // Input: The dmabuf_fd
+ __u64 size; // Output: The size of the dmabuf
+}VB_DMABUF_SIZE_CMD_S;
+
+#define MMZ_VB_IOC_MAGIC 'M'
+#define MMZ_VB_IOCTL_GET_BLOCK _IOWR(MMZ_VB_IOC_MAGIC, 0x0, struct esVB_GET_BLOCK_CMD_S)
+#define MMZ_VB_IOCTL_SET_CFG _IOWR(MMZ_VB_IOC_MAGIC, 0x1, struct esVB_SET_CFG_CMD_S)
+#define MMZ_VB_IOCTL_GET_CFG _IOWR(MMZ_VB_IOC_MAGIC, 0x2, struct esVB_GET_CFG_CMD_S)
+#define MMZ_VB_IOCTL_INIT_CFG _IOWR(MMZ_VB_IOC_MAGIC, 0x3, struct esVB_INIT_CFG_CMD_S)
+#define MMZ_VB_IOCTL_UNINIT_CFG _IOWR(MMZ_VB_IOC_MAGIC, 0x4, struct esVB_UNINIT_CFG_CMD_S)
+#define MMZ_VB_IOCTL_CREATE_POOL _IOWR(MMZ_VB_IOC_MAGIC, 0x5, struct esVB_CREATE_POOL_CMD_S)
+#define MMZ_VB_IOCTL_DESTORY_POOL _IOWR(MMZ_VB_IOC_MAGIC, 0x6, struct esVB_DESTORY_POOL_CMD_S)
+#define MMZ_VB_IOCTL_POOL_SIZE _IOR(MMZ_VB_IOC_MAGIC, 0x7, struct esVB_GET_POOLSIZE_CMD_S)
+#define MMZ_VB_IOCTL_FLUSH_POOL _IOW(MMZ_VB_IOC_MAGIC, 0x8, struct esVB_FLUSH_POOL_CMD_S)
+#define MMZ_VB_IOCTL_BLOCK_TO_POOL _IOR(MMZ_VB_IOC_MAGIC, 0x9, struct esVB_BLOCK_TO_POOL_CMD_S)
+#define MMZ_VB_IOCTL_GET_BLOCK_OFFSET _IOR(MMZ_VB_IOC_MAGIC, 0xa, struct esVB_GET_BLOCKOFFSET_CMD_S)
+#define MMZ_VB_IOCTL_SPLIT_DMABUF _IOWR(MMZ_VB_IOC_MAGIC, 0xb, struct esVB_SPLIT_DMABUF_CMD_S)
+#define MMZ_VB_IOCTL_DMABUF_REFCOUNT _IOR(MMZ_VB_IOC_MAGIC, 0xc, struct esVB_DMABUF_REFCOUNT_CMD_S)
+#define MMZ_VB_IOCTL_RETRIEVE_MEM_NODE _IOR(MMZ_VB_IOC_MAGIC, 0xd, struct esVB_RETRIEVE_MEM_NODE_CMD_S)
+#define MMZ_VB_IOCTL_DMABUF_SIZE _IOR(MMZ_VB_IOC_MAGIC, 0xe, struct esVB_DMABUF_SIZE_CMD_S)
+
+#endif
--
2.47.0