1 /*
2 * FreeRTOS Kernel V11.1.0
3 * Copyright (C) 2015-2019 Cadence Design Systems, Inc.
4 * Copyright (C) 2021 Amazon.com, Inc. or its affiliates. All Rights Reserved.
5 *
6 * SPDX-License-Identifier: MIT
7 *
8 * Permission is hereby granted, free of charge, to any person obtaining a copy of
9 * this software and associated documentation files (the "Software"), to deal in
10 * the Software without restriction, including without limitation the rights to
11 * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
12 * the Software, and to permit persons to whom the Software is furnished to do so,
13 * subject to the following conditions:
14 *
15 * The above copyright notice and this permission notice shall be included in all
16 * copies or substantial portions of the Software.
17 *
18 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
19 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
20 * FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
21 * COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
22 * IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
23 * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
24 *
25 * https://www.FreeRTOS.org
26 * https://github.com/FreeRTOS
27 *
28 */
29
30 #include "FreeRTOS.h"
31
32 #if XT_USE_THREAD_SAFE_CLIB
33
34 #if XSHAL_CLIB == XTHAL_CLIB_XCLIB
35
36 #include <errno.h>
37 #include <sys/reent.h>
38
39 #include "semphr.h"
40
41 typedef SemaphoreHandle_t _Rmtx;
42
43 //-----------------------------------------------------------------------------
44 // Override this and set to nonzero to enable locking.
45 //-----------------------------------------------------------------------------
46 int32_t _xclib_use_mt = 1;
47
48
49 //-----------------------------------------------------------------------------
50 // Init lock.
51 //-----------------------------------------------------------------------------
52 void
_Mtxinit(_Rmtx * mtx)53 _Mtxinit(_Rmtx * mtx)
54 {
55 *mtx = xSemaphoreCreateRecursiveMutex();
56 }
57
58 //-----------------------------------------------------------------------------
59 // Destroy lock.
60 //-----------------------------------------------------------------------------
61 void
_Mtxdst(_Rmtx * mtx)62 _Mtxdst(_Rmtx * mtx)
63 {
64 if ((mtx != NULL) && (*mtx != NULL)) {
65 vSemaphoreDelete(*mtx);
66 }
67 }
68
69 //-----------------------------------------------------------------------------
70 // Lock.
71 //-----------------------------------------------------------------------------
72 void
_Mtxlock(_Rmtx * mtx)73 _Mtxlock(_Rmtx * mtx)
74 {
75 if ((mtx != NULL) && (*mtx != NULL)) {
76 xSemaphoreTakeRecursive(*mtx, portMAX_DELAY);
77 }
78 }
79
80 //-----------------------------------------------------------------------------
81 // Unlock.
82 //-----------------------------------------------------------------------------
83 void
_Mtxunlock(_Rmtx * mtx)84 _Mtxunlock(_Rmtx * mtx)
85 {
86 if ((mtx != NULL) && (*mtx != NULL)) {
87 xSemaphoreGiveRecursive(*mtx);
88 }
89 }
90
91 //-----------------------------------------------------------------------------
92 // Called by malloc() to allocate blocks of memory from the heap.
93 //-----------------------------------------------------------------------------
94 void *
_sbrk_r(struct _reent * reent,int32_t incr)95 _sbrk_r (struct _reent * reent, int32_t incr)
96 {
97 extern char _end;
98 extern char _heap_sentry;
99 static char * _heap_sentry_ptr = &_heap_sentry;
100 static char * heap_ptr;
101 char * base;
102
103 if (!heap_ptr)
104 heap_ptr = (char *) &_end;
105
106 base = heap_ptr;
107 if (heap_ptr + incr >= _heap_sentry_ptr) {
108 reent->_errno = ENOMEM;
109 return (char *) -1;
110 }
111
112 heap_ptr += incr;
113 return base;
114 }
115
116 //-----------------------------------------------------------------------------
117 // Global initialization for C library.
118 //-----------------------------------------------------------------------------
119 void
vPortClibInit(void)120 vPortClibInit(void)
121 {
122 }
123
124 //-----------------------------------------------------------------------------
125 // Per-thread cleanup stub provided for linking, does nothing.
126 //-----------------------------------------------------------------------------
127 void
_reclaim_reent(void * ptr)128 _reclaim_reent(void * ptr)
129 {
130 }
131
132 #endif /* XSHAL_CLIB == XTHAL_CLIB_XCLIB */
133
134 #if XSHAL_CLIB == XTHAL_CLIB_NEWLIB
135
136 #include <errno.h>
137 #include <malloc.h>
138 #include <stdio.h>
139 #include <stdlib.h>
140 #include <string.h>
141
142 #include "semphr.h"
143
144 static SemaphoreHandle_t xClibMutex;
145 static uint32_t ulClibInitDone = 0;
146
147 //-----------------------------------------------------------------------------
148 // Get C library lock.
149 //-----------------------------------------------------------------------------
150 void
__malloc_lock(struct _reent * ptr)151 __malloc_lock(struct _reent * ptr)
152 {
153 if (!ulClibInitDone)
154 return;
155
156 xSemaphoreTakeRecursive(xClibMutex, portMAX_DELAY);
157 }
158
159 //-----------------------------------------------------------------------------
160 // Release C library lock.
161 //-----------------------------------------------------------------------------
162 void
__malloc_unlock(struct _reent * ptr)163 __malloc_unlock(struct _reent * ptr)
164 {
165 if (!ulClibInitDone)
166 return;
167
168 xSemaphoreGiveRecursive(xClibMutex);
169 }
170
171 //-----------------------------------------------------------------------------
172 // Lock for environment. Since we have only one global lock we can just call
173 // the malloc() lock function.
174 //-----------------------------------------------------------------------------
175 void
__env_lock(struct _reent * ptr)176 __env_lock(struct _reent * ptr)
177 {
178 __malloc_lock(ptr);
179 }
180
181
182 //-----------------------------------------------------------------------------
183 // Unlock environment.
184 //-----------------------------------------------------------------------------
185 void
__env_unlock(struct _reent * ptr)186 __env_unlock(struct _reent * ptr)
187 {
188 __malloc_unlock(ptr);
189 }
190
191 //-----------------------------------------------------------------------------
192 // Called by malloc() to allocate blocks of memory from the heap.
193 //-----------------------------------------------------------------------------
194 void *
_sbrk_r(struct _reent * reent,int32_t incr)195 _sbrk_r (struct _reent * reent, int32_t incr)
196 {
197 extern char _end;
198 extern char _heap_sentry;
199 static char * _heap_sentry_ptr = &_heap_sentry;
200 static char * heap_ptr;
201 char * base;
202
203 if (!heap_ptr)
204 heap_ptr = (char *) &_end;
205
206 base = heap_ptr;
207 if (heap_ptr + incr >= _heap_sentry_ptr) {
208 reent->_errno = ENOMEM;
209 return (char *) -1;
210 }
211
212 heap_ptr += incr;
213 return base;
214 }
215
216 //-----------------------------------------------------------------------------
217 // Global initialization for C library.
218 //-----------------------------------------------------------------------------
219 void
vPortClibInit(void)220 vPortClibInit(void)
221 {
222 configASSERT(!ulClibInitDone);
223
224 xClibMutex = xSemaphoreCreateRecursiveMutex();
225 ulClibInitDone = 1;
226 }
227
228 #endif /* XSHAL_CLIB == XTHAL_CLIB_NEWLIB */
229
230 #endif /* XT_USE_THREAD_SAFE_CLIB */
231