xref: /aosp_15_r20/external/fastrpc/src/apps_mem_imp.c (revision 418b791d679beb2078b579a3b6936cf330c41799)
1*418b791dSBob Badour /*
2*418b791dSBob Badour  * Copyright (c) 2019, The Linux Foundation. All rights reserved.
3*418b791dSBob Badour  *
4*418b791dSBob Badour  * Redistribution and use in source and binary forms, with or without
5*418b791dSBob Badour  * modification, are permitted provided that the following conditions are
6*418b791dSBob Badour  * met:
7*418b791dSBob Badour  *    * Redistributions of source code must retain the above copyright
8*418b791dSBob Badour  *      notice, this list of conditions and the following disclaimer.
9*418b791dSBob Badour  *    * Redistributions in binary form must reproduce the above
10*418b791dSBob Badour  *      copyright notice, this list of conditions and the following
11*418b791dSBob Badour  *      disclaimer in the documentation and/or other materials provided
12*418b791dSBob Badour  *      with the distribution.
13*418b791dSBob Badour  *    * Neither the name of The Linux Foundation nor the names of its
14*418b791dSBob Badour  *      contributors may be used to endorse or promote products derived
15*418b791dSBob Badour  *      from this software without specific prior written permission.
16*418b791dSBob Badour  *
17*418b791dSBob Badour  * THIS SOFTWARE IS PROVIDED "AS IS" AND ANY EXPRESS OR IMPLIED
18*418b791dSBob Badour  * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
19*418b791dSBob Badour  * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT
20*418b791dSBob Badour  * ARE DISCLAIMED.  IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS
21*418b791dSBob Badour  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
22*418b791dSBob Badour  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
23*418b791dSBob Badour  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
24*418b791dSBob Badour  * BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
25*418b791dSBob Badour  * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE
26*418b791dSBob Badour  * OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN
27*418b791dSBob Badour  * IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28*418b791dSBob Badour  */
29*418b791dSBob Badour #ifndef VERIFY_PRINT_ERROR
30*418b791dSBob Badour #define VERIFY_PRINT_ERROR
31*418b791dSBob Badour #endif /* VERIFY_PRINT_ERROR */
32*418b791dSBob Badour 
33*418b791dSBob Badour #include <stdio.h>
34*418b791dSBob Badour #include <stdlib.h>
35*418b791dSBob Badour #include <string.h>
36*418b791dSBob Badour #include <pthread.h>
37*418b791dSBob Badour #include <sys/mman.h>
38*418b791dSBob Badour #include "apps_mem.h"
39*418b791dSBob Badour #include "remote64.h"
40*418b791dSBob Badour #include "rpcmem.h"
41*418b791dSBob Badour #include "verify.h"
42*418b791dSBob Badour #include "rpcmem.h"
43*418b791dSBob Badour #include "AEEQList.h"
44*418b791dSBob Badour #include "AEEstd.h"
45*418b791dSBob Badour #include "AEEStdErr.h"
46*418b791dSBob Badour #include "fastrpc_apps_user.h"
47*418b791dSBob Badour #include "platform_libs.h"
48*418b791dSBob Badour 
49*418b791dSBob Badour #define ADSP_MMAP_HEAP_ADDR 4
50*418b791dSBob Badour #define ADSP_MMAP_REMOTE_HEAP_ADDR 8
51*418b791dSBob Badour #define ADSP_MMAP_ADD_PAGES   0x1000
52*418b791dSBob Badour 
53*418b791dSBob Badour static QList memlst;
54*418b791dSBob Badour static pthread_mutex_t memmt;
55*418b791dSBob Badour 
56*418b791dSBob Badour struct mem_info {
57*418b791dSBob Badour    QNode qn;
58*418b791dSBob Badour    uint64 vapps;
59*418b791dSBob Badour    uint64 vadsp;
60*418b791dSBob Badour    int32 size;
61*418b791dSBob Badour    int32 mapped;
62*418b791dSBob Badour };
63*418b791dSBob Badour 
64*418b791dSBob Badour /*
65*418b791dSBob Badour These should be called in some static constructor of the .so that
66*418b791dSBob Badour uses rpcmem.
67*418b791dSBob Badour 
68*418b791dSBob Badour I moved them into fastrpc_apps_user.c because there is no gurantee in
69*418b791dSBob Badour the order of when constructors are called.
70*418b791dSBob Badour */
71*418b791dSBob Badour 
apps_mem_init(void)72*418b791dSBob Badour static int apps_mem_init(void) {
73*418b791dSBob Badour    QList_Ctor(&memlst);
74*418b791dSBob Badour    pthread_mutex_init(&memmt, 0);
75*418b791dSBob Badour    return AEE_SUCCESS;
76*418b791dSBob Badour }
77*418b791dSBob Badour 
apps_mem_deinit(void)78*418b791dSBob Badour void apps_mem_deinit(void) {
79*418b791dSBob Badour    QNode *pn;
80*418b791dSBob Badour    while ((pn = QList_PopZ(&memlst)) != NULL) {
81*418b791dSBob Badour       struct mem_info *mfree = STD_RECOVER_REC(struct mem_info, qn, pn);
82*418b791dSBob Badour 
83*418b791dSBob Badour       if (mfree->vapps) {
84*418b791dSBob Badour          if(mfree->mapped) {
85*418b791dSBob Badour             munmap((void*)(uintptr_t)mfree->vapps, mfree->size);
86*418b791dSBob Badour          } else {
87*418b791dSBob Badour             rpcmem_free_internal((void *)(uintptr_t)mfree->vapps);
88*418b791dSBob Badour          }
89*418b791dSBob Badour       }
90*418b791dSBob Badour       free(mfree);
91*418b791dSBob Badour       mfree = NULL;
92*418b791dSBob Badour    }
93*418b791dSBob Badour    pthread_mutex_destroy(&memmt);
94*418b791dSBob Badour }
95*418b791dSBob Badour PL_DEFINE(apps_mem, apps_mem_init, apps_mem_deinit);
96*418b791dSBob Badour 
__QAIC_IMPL(apps_mem_request_map64)97*418b791dSBob Badour __QAIC_IMPL_EXPORT int __QAIC_IMPL(apps_mem_request_map64)(int heapid, uint32 lflags, uint32 rflags, uint64 vin, int64 len, uint64* vapps, uint64* vadsp) __QAIC_IMPL_ATTRIBUTE {
98*418b791dSBob Badour    struct mem_info *minfo = 0;
99*418b791dSBob Badour    int nErr = 0;
100*418b791dSBob Badour    void* buf = 0;
101*418b791dSBob Badour    uint64_t pbuf;
102*418b791dSBob Badour    int fd = -1;
103*418b791dSBob Badour 
104*418b791dSBob Badour    (void)vin;
105*418b791dSBob Badour    VERIFYC(NULL != (minfo = malloc(sizeof(*minfo))), AEE_ENOMEMORY);
106*418b791dSBob Badour    QNode_CtorZ(&minfo->qn);
107*418b791dSBob Badour    *vadsp = 0;
108*418b791dSBob Badour    if (rflags == ADSP_MMAP_HEAP_ADDR || rflags == ADSP_MMAP_REMOTE_HEAP_ADDR) {
109*418b791dSBob Badour 	VERIFY(AEE_SUCCESS == (nErr = remote_mmap64(-1, rflags, 0, len, (uint64_t*)vadsp)));
110*418b791dSBob Badour 	*vapps = 0;
111*418b791dSBob Badour 	minfo->vapps = 0;
112*418b791dSBob Badour    } else {
113*418b791dSBob Badour 	if ((rflags != ADSP_MMAP_ADD_PAGES) ||
114*418b791dSBob Badour 		((rflags == ADSP_MMAP_ADD_PAGES) && !is_kernel_alloc_supported(-1, -1))) {
115*418b791dSBob Badour 		VERIFYC(NULL != (buf = rpcmem_alloc_internal(heapid, lflags, len)), AEE_ENORPCMEMORY);
116*418b791dSBob Badour 		fd = rpcmem_to_fd_internal(buf);
117*418b791dSBob Badour 		VERIFYC(fd > 0, AEE_EINVALIDFD);
118*418b791dSBob Badour 	}
119*418b791dSBob Badour 	VERIFY(AEE_SUCCESS == (nErr = remote_mmap64(fd, rflags, (uint64_t)buf, len, (uint64_t*)vadsp)));
120*418b791dSBob Badour 	pbuf = (uint64_t)buf;
121*418b791dSBob Badour 	*vapps = pbuf;
122*418b791dSBob Badour 	minfo->vapps = *vapps;
123*418b791dSBob Badour    }
124*418b791dSBob Badour    minfo->vadsp = *vadsp;
125*418b791dSBob Badour    minfo->size = len;
126*418b791dSBob Badour    minfo->mapped = 0;
127*418b791dSBob Badour    pthread_mutex_lock(&memmt);
128*418b791dSBob Badour    QList_AppendNode(&memlst, &minfo->qn);
129*418b791dSBob Badour    pthread_mutex_unlock(&memmt);
130*418b791dSBob Badour bail:
131*418b791dSBob Badour    if(nErr) {
132*418b791dSBob Badour       if(buf) {
133*418b791dSBob Badour          rpcmem_free_internal(buf);
134*418b791dSBob Badour          buf = NULL;
135*418b791dSBob Badour       }
136*418b791dSBob Badour       if(minfo) {
137*418b791dSBob Badour          free(minfo);
138*418b791dSBob Badour          minfo = NULL;
139*418b791dSBob Badour       }
140*418b791dSBob Badour       VERIFY_EPRINTF("Error %x: apps_mem_request_mmap64 failed\n", nErr);
141*418b791dSBob Badour    }
142*418b791dSBob Badour    return nErr;
143*418b791dSBob Badour }
144*418b791dSBob Badour 
__QAIC_IMPL(apps_mem_request_map)145*418b791dSBob Badour __QAIC_IMPL_EXPORT int __QAIC_IMPL(apps_mem_request_map)(int heapid, uint32 lflags, uint32 rflags, uint32 vin, int32 len, uint32* vapps, uint32* vadsp) __QAIC_IMPL_ATTRIBUTE {
146*418b791dSBob Badour    uint64 vin1, vapps1, vadsp1;
147*418b791dSBob Badour    int64 len1;
148*418b791dSBob Badour    int nErr = AEE_SUCCESS;
149*418b791dSBob Badour    vin1 = (uint64)vin;
150*418b791dSBob Badour    len1 = (int64)len;
151*418b791dSBob Badour    nErr = apps_mem_request_map64(heapid, lflags, rflags, vin1, len1, &vapps1, &vadsp1);
152*418b791dSBob Badour    *vapps = (uint32)vapps1;
153*418b791dSBob Badour    *vadsp = (uint32)vadsp1;
154*418b791dSBob Badour    if(nErr != AEE_SUCCESS) {
155*418b791dSBob Badour 	VERIFY_EPRINTF("Error %x: apps_mem_request_map failed\n", nErr);
156*418b791dSBob Badour    }
157*418b791dSBob Badour    return nErr;
158*418b791dSBob Badour }
159*418b791dSBob Badour 
__QAIC_IMPL(apps_mem_request_unmap64)160*418b791dSBob Badour __QAIC_IMPL_EXPORT int __QAIC_IMPL(apps_mem_request_unmap64)(uint64 vadsp, int64 len) __QAIC_IMPL_ATTRIBUTE {
161*418b791dSBob Badour    int nErr = AEE_SUCCESS;
162*418b791dSBob Badour    struct mem_info *minfo, *mfree = 0;
163*418b791dSBob Badour    QNode *pn, *pnn;
164*418b791dSBob Badour    VERIFY(0 == (nErr = remote_munmap64((uint64_t)vadsp, len)));
165*418b791dSBob Badour    pthread_mutex_lock(&memmt);
166*418b791dSBob Badour    QLIST_NEXTSAFE_FOR_ALL(&memlst, pn, pnn) {
167*418b791dSBob Badour       minfo = STD_RECOVER_REC(struct mem_info, qn, pn);
168*418b791dSBob Badour       if(minfo->vadsp == vadsp) {
169*418b791dSBob Badour          mfree = minfo;
170*418b791dSBob Badour          QNode_Dequeue(&minfo->qn);
171*418b791dSBob Badour          break;
172*418b791dSBob Badour       }
173*418b791dSBob Badour    }
174*418b791dSBob Badour    pthread_mutex_unlock(&memmt);
175*418b791dSBob Badour    VERIFYC(mfree, AEE_ENOSUCHMAP);
176*418b791dSBob Badour    if(mfree->mapped) {
177*418b791dSBob Badour       munmap((void*)(uintptr_t)mfree->vapps, mfree->size);
178*418b791dSBob Badour    } else {
179*418b791dSBob Badour       if (mfree->vapps)
180*418b791dSBob Badour          rpcmem_free_internal((void *)(uintptr_t)mfree->vapps);
181*418b791dSBob Badour    }
182*418b791dSBob Badour    free(mfree);
183*418b791dSBob Badour    mfree = NULL;
184*418b791dSBob Badour bail:
185*418b791dSBob Badour    if(nErr != AEE_SUCCESS) {
186*418b791dSBob Badour 	VERIFY_EPRINTF("Error %x: apps_mem_request_unmap64 failed\n", nErr);
187*418b791dSBob Badour    }
188*418b791dSBob Badour    return nErr;
189*418b791dSBob Badour }
190*418b791dSBob Badour 
__QAIC_IMPL(apps_mem_request_unmap)191*418b791dSBob Badour __QAIC_IMPL_EXPORT int __QAIC_IMPL(apps_mem_request_unmap)(uint32 vadsp, int32 len) __QAIC_IMPL_ATTRIBUTE {
192*418b791dSBob Badour    uint64 vadsp1 = (uint64)vadsp;
193*418b791dSBob Badour    int64 len1 = (int64)len;
194*418b791dSBob Badour    int nErr = apps_mem_request_unmap64(vadsp1, len1);
195*418b791dSBob Badour    if(nErr != AEE_SUCCESS) {
196*418b791dSBob Badour 	VERIFY_EPRINTF("Error %x: apps_mem_request_unmap failed\n", nErr);
197*418b791dSBob Badour    }
198*418b791dSBob Badour    return nErr;
199*418b791dSBob Badour }
200*418b791dSBob Badour 
__QAIC_IMPL(apps_mem_share_map)201*418b791dSBob Badour __QAIC_IMPL_EXPORT int __QAIC_IMPL(apps_mem_share_map)(int fd, int size, uint64* vapps, uint64* vadsp) __QAIC_IMPL_ATTRIBUTE {
202*418b791dSBob Badour    struct mem_info *minfo = 0;
203*418b791dSBob Badour    int nErr = AEE_SUCCESS;
204*418b791dSBob Badour    void* buf = 0;
205*418b791dSBob Badour    uint64_t pbuf;
206*418b791dSBob Badour    VERIFYC(0 != (minfo = malloc(sizeof(*minfo))), AEE_ENOMEMORY);
207*418b791dSBob Badour    QNode_CtorZ(&minfo->qn);
208*418b791dSBob Badour    VERIFYC(fd > 0, AEE_EINVALIDFD);
209*418b791dSBob Badour    *vadsp = 0;
210*418b791dSBob Badour    VERIFYC(MAP_FAILED != (buf = (void *)mmap(NULL, size,
211*418b791dSBob Badour                            PROT_READ|PROT_WRITE, MAP_SHARED, fd, 0)), AEE_EMMAP);
212*418b791dSBob Badour    VERIFY(AEE_SUCCESS == (nErr = remote_mmap64(fd, 0, (uint64_t)buf, size, (uint64_t*)vadsp)));
213*418b791dSBob Badour    pbuf = (uint64_t)buf;
214*418b791dSBob Badour    *vapps = pbuf;
215*418b791dSBob Badour    minfo->vapps = *vapps;
216*418b791dSBob Badour    minfo->vadsp = *vadsp;
217*418b791dSBob Badour    minfo->size = size;
218*418b791dSBob Badour    minfo->mapped = 1;
219*418b791dSBob Badour    pthread_mutex_lock(&memmt);
220*418b791dSBob Badour    QList_AppendNode(&memlst, &minfo->qn);
221*418b791dSBob Badour    pthread_mutex_unlock(&memmt);
222*418b791dSBob Badour bail:
223*418b791dSBob Badour    if(nErr) {
224*418b791dSBob Badour       if(buf) {
225*418b791dSBob Badour          munmap(buf, size);
226*418b791dSBob Badour       }
227*418b791dSBob Badour       if(minfo) {
228*418b791dSBob Badour          free(minfo);
229*418b791dSBob Badour          minfo = NULL;
230*418b791dSBob Badour       }
231*418b791dSBob Badour       VERIFY_EPRINTF("Error %x: apps_mem_share_map failed\n", nErr);
232*418b791dSBob Badour    }
233*418b791dSBob Badour    return nErr;
234*418b791dSBob Badour }
235*418b791dSBob Badour 
__QAIC_IMPL(apps_mem_share_unmap)236*418b791dSBob Badour __QAIC_IMPL_EXPORT int __QAIC_IMPL(apps_mem_share_unmap)(uint64 vadsp, int size) __QAIC_IMPL_ATTRIBUTE {
237*418b791dSBob Badour    int64 len1 = (int64)size;
238*418b791dSBob Badour    int nErr = AEE_SUCCESS;
239*418b791dSBob Badour    nErr = apps_mem_request_unmap64(vadsp, len1);
240*418b791dSBob Badour    if(nErr != AEE_SUCCESS) {
241*418b791dSBob Badour 	VERIFY_EPRINTF("Error %x: apps_mem_share_unmap failed\n", nErr);
242*418b791dSBob Badour    }
243*418b791dSBob Badour    return nErr;
244*418b791dSBob Badour }
245