Merge branch 'newlib'
This commit is contained in:
		
						commit
						86188c01fd
					
				
					 133 changed files with 23672 additions and 578 deletions
				
			
		|  | @ -1,387 +0,0 @@ | |||
| /*
 | ||||
|     FreeRTOS V7.5.2 - Copyright (C) 2013 Real Time Engineers Ltd. | ||||
| 
 | ||||
|     VISIT http://www.FreeRTOS.org TO ENSURE YOU ARE USING THE LATEST VERSION.
 | ||||
| 
 | ||||
|     *************************************************************************** | ||||
|      *                                                                       * | ||||
|      *    FreeRTOS provides completely free yet professionally developed,    * | ||||
|      *    robust, strictly quality controlled, supported, and cross          * | ||||
|      *    platform software that has become a de facto standard.             * | ||||
|      *                                                                       * | ||||
|      *    Help yourself get started quickly and support the FreeRTOS         * | ||||
|      *    project by purchasing a FreeRTOS tutorial book, reference          * | ||||
|      *    manual, or both from: http://www.FreeRTOS.org/Documentation        *
 | ||||
|      *                                                                       * | ||||
|      *    Thank you!                                                         * | ||||
|      *                                                                       * | ||||
|     *************************************************************************** | ||||
| 
 | ||||
|     This file is part of the FreeRTOS distribution. | ||||
| 
 | ||||
|     FreeRTOS is free software; you can redistribute it and/or modify it under | ||||
|     the terms of the GNU General Public License (version 2) as published by the | ||||
|     Free Software Foundation >>!AND MODIFIED BY!<< the FreeRTOS exception. | ||||
| 
 | ||||
|     >>! NOTE: The modification to the GPL is included to allow you to distribute | ||||
|     >>! a combined work that includes FreeRTOS without being obliged to provide | ||||
|     >>! the source code for proprietary components outside of the FreeRTOS | ||||
|     >>! kernel. | ||||
| 
 | ||||
|     FreeRTOS is distributed in the hope that it will be useful, but WITHOUT ANY | ||||
|     WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS | ||||
|     FOR A PARTICULAR PURPOSE.  Full license text is available from the following | ||||
|     link: http://www.freertos.org/a00114.html
 | ||||
| 
 | ||||
|     1 tab == 4 spaces! | ||||
| 
 | ||||
|     *************************************************************************** | ||||
|      *                                                                       * | ||||
|      *    Having a problem?  Start by reading the FAQ "My application does   * | ||||
|      *    not run, what could be wrong?"                                     * | ||||
|      *                                                                       * | ||||
|      *    http://www.FreeRTOS.org/FAQHelp.html                               *
 | ||||
|      *                                                                       * | ||||
|     *************************************************************************** | ||||
| 
 | ||||
|     http://www.FreeRTOS.org - Documentation, books, training, latest versions,
 | ||||
|     license and Real Time Engineers Ltd. contact details. | ||||
| 
 | ||||
|     http://www.FreeRTOS.org/plus - A selection of FreeRTOS ecosystem products,
 | ||||
|     including FreeRTOS+Trace - an indispensable productivity tool, a DOS | ||||
|     compatible FAT file system, and our tiny thread aware UDP/IP stack. | ||||
| 
 | ||||
|     http://www.OpenRTOS.com - Real Time Engineers ltd license FreeRTOS to High
 | ||||
|     Integrity Systems to sell under the OpenRTOS brand.  Low cost OpenRTOS | ||||
|     licenses offer ticketed support, indemnification and middleware. | ||||
| 
 | ||||
|     http://www.SafeRTOS.com - High Integrity Systems also provide a safety
 | ||||
|     engineered and independently SIL3 certified version for use in safety and | ||||
|     mission critical applications that require provable dependability. | ||||
| 
 | ||||
|     1 tab == 4 spaces! | ||||
| */ | ||||
| 
 | ||||
| /*
 | ||||
|  * A sample implementation of pvPortMalloc() and vPortFree() that combines  | ||||
|  * (coalescences) adjacent memory blocks as they are freed, and in so doing  | ||||
|  * limits memory fragmentation. | ||||
|  * | ||||
|  * See heap_1.c, heap_2.c and heap_3.c for alternative implementations, and the  | ||||
|  * memory management pages of http://www.FreeRTOS.org for more information.
 | ||||
|  */ | ||||
| #include <stdlib.h> | ||||
| 
 | ||||
| /* Defining MPU_WRAPPERS_INCLUDED_FROM_API_FILE prevents task.h from redefining
 | ||||
| all the API functions to use the MPU wrappers.  That should only be done when | ||||
| task.h is included from an application file. */ | ||||
| #define MPU_WRAPPERS_INCLUDED_FROM_API_FILE | ||||
| 
 | ||||
| #include "FreeRTOS.h" | ||||
| #include "task.h" | ||||
| 
 | ||||
| #undef MPU_WRAPPERS_INCLUDED_FROM_API_FILE | ||||
| 
 | ||||
| /* Block sizes must not get too small. */ | ||||
| #define heapMINIMUM_BLOCK_SIZE	( ( size_t ) ( heapSTRUCT_SIZE * 2 ) ) | ||||
| 
 | ||||
| /* Assumes 8bit bytes! */ | ||||
| #define heapBITS_PER_BYTE		( ( size_t ) 8 ) | ||||
| 
 | ||||
| /* A few bytes might be lost to byte aligning the heap start address. */ | ||||
| #define heapADJUSTED_HEAP_SIZE	( configTOTAL_HEAP_SIZE - portBYTE_ALIGNMENT ) | ||||
| 
 | ||||
| /* Allocate the memory for the heap. */ | ||||
| static unsigned char ucHeap[ configTOTAL_HEAP_SIZE ]; | ||||
| 
 | ||||
| /* Define the linked list structure.  This is used to link free blocks in order
 | ||||
| of their memory address. */ | ||||
| typedef struct A_BLOCK_LINK | ||||
| { | ||||
| 	struct A_BLOCK_LINK *pxNextFreeBlock;	/*<< The next free block in the list. */ | ||||
| 	size_t xBlockSize;						/*<< The size of the free block. */ | ||||
| } xBlockLink; | ||||
| 
 | ||||
| /*-----------------------------------------------------------*/ | ||||
| 
 | ||||
| /*
 | ||||
|  * Inserts a block of memory that is being freed into the correct position in  | ||||
|  * the list of free memory blocks.  The block being freed will be merged with | ||||
|  * the block in front it and/or the block behind it if the memory blocks are | ||||
|  * adjacent to each other. | ||||
|  */ | ||||
| static void prvInsertBlockIntoFreeList( xBlockLink *pxBlockToInsert ); | ||||
| 
 | ||||
| /*
 | ||||
|  * Called automatically to setup the required heap structures the first time | ||||
|  * pvPortMalloc() is called. | ||||
|  */ | ||||
| static void prvHeapInit( void ); | ||||
| 
 | ||||
| /*-----------------------------------------------------------*/ | ||||
| 
 | ||||
| /* The size of the structure placed at the beginning of each allocated memory
 | ||||
| block must by correctly byte aligned. */ | ||||
| static const unsigned short heapSTRUCT_SIZE	= ( ( sizeof ( xBlockLink ) + ( portBYTE_ALIGNMENT - 1 ) ) & ~portBYTE_ALIGNMENT_MASK ); | ||||
| 
 | ||||
| /* Ensure the pxEnd pointer will end up on the correct byte alignment. */ | ||||
| static const size_t xTotalHeapSize = ( ( size_t ) heapADJUSTED_HEAP_SIZE ) & ( ( size_t ) ~portBYTE_ALIGNMENT_MASK ); | ||||
| 
 | ||||
| /* Create a couple of list links to mark the start and end of the list. */ | ||||
| static xBlockLink xStart, *pxEnd = NULL; | ||||
| 
 | ||||
| /* Keeps track of the number of free bytes remaining, but says nothing about
 | ||||
| fragmentation. */ | ||||
| static size_t xFreeBytesRemaining = ( ( size_t ) heapADJUSTED_HEAP_SIZE ) & ( ( size_t ) ~portBYTE_ALIGNMENT_MASK ); | ||||
| 
 | ||||
| /* Gets set to the top bit of an size_t type.  When this bit in the xBlockSize 
 | ||||
| member of an xBlockLink structure is set then the block belongs to the  | ||||
| application.  When the bit is free the block is still part of the free heap | ||||
| space. */ | ||||
| static size_t xBlockAllocatedBit = 0; | ||||
| 
 | ||||
| /*-----------------------------------------------------------*/ | ||||
| 
 | ||||
| void *pvPortMalloc( size_t xWantedSize ) | ||||
| { | ||||
| xBlockLink *pxBlock, *pxPreviousBlock, *pxNewBlockLink; | ||||
| void *pvReturn = NULL; | ||||
| 
 | ||||
| 	vTaskSuspendAll(); | ||||
| 	{ | ||||
| 		/* If this is the first call to malloc then the heap will require
 | ||||
| 		initialisation to setup the list of free blocks. */ | ||||
| 		if( pxEnd == NULL ) | ||||
| 		{ | ||||
| 			prvHeapInit(); | ||||
| 		} | ||||
| 
 | ||||
| 		/* Check the requested block size is not so large that the top bit is
 | ||||
| 		set.  The top bit of the block size member of the xBlockLink structure  | ||||
| 		is used to determine who owns the block - the application or the | ||||
| 		kernel, so it must be free. */ | ||||
| 		if( ( xWantedSize & xBlockAllocatedBit ) == 0 ) | ||||
| 		{ | ||||
| 			/* The wanted size is increased so it can contain a xBlockLink
 | ||||
| 			structure in addition to the requested amount of bytes. */ | ||||
| 			if( xWantedSize > 0 ) | ||||
| 			{ | ||||
| 				xWantedSize += heapSTRUCT_SIZE; | ||||
| 
 | ||||
| 				/* Ensure that blocks are always aligned to the required number 
 | ||||
| 				of bytes. */ | ||||
| 				if( ( xWantedSize & portBYTE_ALIGNMENT_MASK ) != 0x00 ) | ||||
| 				{ | ||||
| 					/* Byte alignment required. */ | ||||
| 					xWantedSize += ( portBYTE_ALIGNMENT - ( xWantedSize & portBYTE_ALIGNMENT_MASK ) ); | ||||
| 				} | ||||
| 			} | ||||
| 
 | ||||
| 			if( ( xWantedSize > 0 ) && ( xWantedSize <= xFreeBytesRemaining ) ) | ||||
| 			{ | ||||
| 				/* Traverse the list from the start	(lowest address) block until 
 | ||||
| 				one	of adequate size is found. */ | ||||
| 				pxPreviousBlock = &xStart; | ||||
| 				pxBlock = xStart.pxNextFreeBlock; | ||||
| 				while( ( pxBlock->xBlockSize < xWantedSize ) && ( pxBlock->pxNextFreeBlock != NULL ) ) | ||||
| 				{ | ||||
| 					pxPreviousBlock = pxBlock; | ||||
| 					pxBlock = pxBlock->pxNextFreeBlock; | ||||
| 				} | ||||
| 
 | ||||
| 				/* If the end marker was reached then a block of adequate size 
 | ||||
| 				was	not found. */ | ||||
| 				if( pxBlock != pxEnd ) | ||||
| 				{ | ||||
| 					/* Return the memory space pointed to - jumping over the 
 | ||||
| 					xBlockLink structure at its start. */ | ||||
| 					pvReturn = ( void * ) ( ( ( unsigned char * ) pxPreviousBlock->pxNextFreeBlock ) + heapSTRUCT_SIZE ); | ||||
| 
 | ||||
| 					/* This block is being returned for use so must be taken out 
 | ||||
| 					of the list of free blocks. */ | ||||
| 					pxPreviousBlock->pxNextFreeBlock = pxBlock->pxNextFreeBlock; | ||||
| 
 | ||||
| 					/* If the block is larger than required it can be split into 
 | ||||
| 					two. */ | ||||
| 					if( ( pxBlock->xBlockSize - xWantedSize ) > heapMINIMUM_BLOCK_SIZE ) | ||||
| 					{ | ||||
| 						/* This block is to be split into two.  Create a new 
 | ||||
| 						block following the number of bytes requested. The void  | ||||
| 						cast is used to prevent byte alignment warnings from the  | ||||
| 						compiler. */ | ||||
| 						pxNewBlockLink = ( void * ) ( ( ( unsigned char * ) pxBlock ) + xWantedSize ); | ||||
| 
 | ||||
| 						/* Calculate the sizes of two blocks split from the 
 | ||||
| 						single block. */ | ||||
| 						pxNewBlockLink->xBlockSize = pxBlock->xBlockSize - xWantedSize; | ||||
| 						pxBlock->xBlockSize = xWantedSize; | ||||
| 
 | ||||
| 						/* Insert the new block into the list of free blocks. */ | ||||
| 						prvInsertBlockIntoFreeList( ( pxNewBlockLink ) ); | ||||
| 					} | ||||
| 
 | ||||
| 					xFreeBytesRemaining -= pxBlock->xBlockSize; | ||||
| 
 | ||||
| 					/* The block is being returned - it is allocated and owned
 | ||||
| 					by the application and has no "next" block. */ | ||||
| 					pxBlock->xBlockSize |= xBlockAllocatedBit; | ||||
| 					pxBlock->pxNextFreeBlock = NULL; | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	xTaskResumeAll(); | ||||
| 
 | ||||
| 	#if( configUSE_MALLOC_FAILED_HOOK == 1 ) | ||||
| 	{ | ||||
| 		if( pvReturn == NULL ) | ||||
| 		{ | ||||
| 			extern void vApplicationMallocFailedHook( void ); | ||||
| 			vApplicationMallocFailedHook(); | ||||
| 		} | ||||
| 	} | ||||
| 	#endif | ||||
| 
 | ||||
| 	return pvReturn; | ||||
| } | ||||
| /*-----------------------------------------------------------*/ | ||||
| 
 | ||||
| void vPortFree( void *pv ) | ||||
| { | ||||
| unsigned char *puc = ( unsigned char * ) pv; | ||||
| xBlockLink *pxLink; | ||||
| 
 | ||||
| 	if( pv != NULL ) | ||||
| 	{ | ||||
| 		/* The memory being freed will have an xBlockLink structure immediately
 | ||||
| 		before it. */ | ||||
| 		puc -= heapSTRUCT_SIZE; | ||||
| 
 | ||||
| 		/* This casting is to keep the compiler from issuing warnings. */ | ||||
| 		pxLink = ( void * ) puc; | ||||
| 
 | ||||
| 		/* Check the block is actually allocated. */ | ||||
| 		configASSERT( ( pxLink->xBlockSize & xBlockAllocatedBit ) != 0 ); | ||||
| 		configASSERT( pxLink->pxNextFreeBlock == NULL ); | ||||
| 		 | ||||
| 		if( ( pxLink->xBlockSize & xBlockAllocatedBit ) != 0 ) | ||||
| 		{ | ||||
| 			if( pxLink->pxNextFreeBlock == NULL ) | ||||
| 			{ | ||||
| 				/* The block is being returned to the heap - it is no longer
 | ||||
| 				allocated. */ | ||||
| 				pxLink->xBlockSize &= ~xBlockAllocatedBit; | ||||
| 
 | ||||
| 				vTaskSuspendAll(); | ||||
| 				{ | ||||
| 					/* Add this block to the list of free blocks. */ | ||||
| 					xFreeBytesRemaining += pxLink->xBlockSize; | ||||
| 					prvInsertBlockIntoFreeList( ( ( xBlockLink * ) pxLink ) ); | ||||
| 				} | ||||
| 				xTaskResumeAll(); | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| /*-----------------------------------------------------------*/ | ||||
| 
 | ||||
| size_t xPortGetFreeHeapSize( void ) | ||||
| { | ||||
| 	return xFreeBytesRemaining; | ||||
| } | ||||
| /*-----------------------------------------------------------*/ | ||||
| 
 | ||||
| void vPortInitialiseBlocks( void ) | ||||
| { | ||||
| 	/* This just exists to keep the linker quiet. */ | ||||
| } | ||||
| /*-----------------------------------------------------------*/ | ||||
| 
 | ||||
| static void prvHeapInit( void ) | ||||
| { | ||||
| xBlockLink *pxFirstFreeBlock; | ||||
| unsigned char *pucHeapEnd, *pucAlignedHeap; | ||||
| 
 | ||||
| 	/* Ensure the heap starts on a correctly aligned boundary. */ | ||||
| 	pucAlignedHeap = ( unsigned char * ) ( ( ( portPOINTER_SIZE_TYPE ) &ucHeap[ portBYTE_ALIGNMENT ] ) & ( ( portPOINTER_SIZE_TYPE ) ~portBYTE_ALIGNMENT_MASK ) ); | ||||
| 
 | ||||
| 	/* xStart is used to hold a pointer to the first item in the list of free
 | ||||
| 	blocks.  The void cast is used to prevent compiler warnings. */ | ||||
| 	xStart.pxNextFreeBlock = ( void * ) pucAlignedHeap; | ||||
| 	xStart.xBlockSize = ( size_t ) 0; | ||||
| 
 | ||||
| 	/* pxEnd is used to mark the end of the list of free blocks and is inserted
 | ||||
| 	at the end of the heap space. */ | ||||
| 	pucHeapEnd = pucAlignedHeap + xTotalHeapSize; | ||||
| 	pucHeapEnd -= heapSTRUCT_SIZE; | ||||
| 	pxEnd = ( void * ) pucHeapEnd; | ||||
| 	configASSERT( ( ( ( unsigned long ) pxEnd ) & ( ( unsigned long ) portBYTE_ALIGNMENT_MASK ) ) == 0UL ); | ||||
| 	pxEnd->xBlockSize = 0; | ||||
| 	pxEnd->pxNextFreeBlock = NULL; | ||||
| 
 | ||||
| 	/* To start with there is a single free block that is sized to take up the
 | ||||
| 	entire heap space, minus the space taken by pxEnd. */ | ||||
| 	pxFirstFreeBlock = ( void * ) pucAlignedHeap; | ||||
| 	pxFirstFreeBlock->xBlockSize = xTotalHeapSize - heapSTRUCT_SIZE; | ||||
| 	pxFirstFreeBlock->pxNextFreeBlock = pxEnd; | ||||
| 
 | ||||
| 	/* The heap now contains pxEnd. */ | ||||
| 	xFreeBytesRemaining -= heapSTRUCT_SIZE; | ||||
| 
 | ||||
| 	/* Work out the position of the top bit in a size_t variable. */ | ||||
| 	xBlockAllocatedBit = ( ( size_t ) 1 ) << ( ( sizeof( size_t ) * heapBITS_PER_BYTE ) - 1 ); | ||||
| } | ||||
| /*-----------------------------------------------------------*/ | ||||
| 
 | ||||
| static void prvInsertBlockIntoFreeList( xBlockLink *pxBlockToInsert ) | ||||
| { | ||||
| xBlockLink *pxIterator; | ||||
| unsigned char *puc; | ||||
| 
 | ||||
| 	/* Iterate through the list until a block is found that has a higher address
 | ||||
| 	than the block being inserted. */ | ||||
| 	for( pxIterator = &xStart; pxIterator->pxNextFreeBlock < pxBlockToInsert; pxIterator = pxIterator->pxNextFreeBlock ) | ||||
| 	{ | ||||
| 		/* Nothing to do here, just iterate to the right position. */ | ||||
| 	} | ||||
| 
 | ||||
| 	/* Do the block being inserted, and the block it is being inserted after
 | ||||
| 	make a contiguous block of memory? */	 | ||||
| 	puc = ( unsigned char * ) pxIterator; | ||||
| 	if( ( puc + pxIterator->xBlockSize ) == ( unsigned char * ) pxBlockToInsert ) | ||||
| 	{ | ||||
| 		pxIterator->xBlockSize += pxBlockToInsert->xBlockSize; | ||||
| 		pxBlockToInsert = pxIterator; | ||||
| 	} | ||||
| 
 | ||||
| 	/* Do the block being inserted, and the block it is being inserted before
 | ||||
| 	make a contiguous block of memory? */ | ||||
| 	puc = ( unsigned char * ) pxBlockToInsert; | ||||
| 	if( ( puc + pxBlockToInsert->xBlockSize ) == ( unsigned char * ) pxIterator->pxNextFreeBlock ) | ||||
| 	{ | ||||
| 		if( pxIterator->pxNextFreeBlock != pxEnd ) | ||||
| 		{ | ||||
| 			/* Form one big block from the two blocks. */ | ||||
| 			pxBlockToInsert->xBlockSize += pxIterator->pxNextFreeBlock->xBlockSize; | ||||
| 			pxBlockToInsert->pxNextFreeBlock = pxIterator->pxNextFreeBlock->pxNextFreeBlock; | ||||
| 		} | ||||
| 		else | ||||
| 		{ | ||||
| 			pxBlockToInsert->pxNextFreeBlock = pxEnd; | ||||
| 		} | ||||
| 	} | ||||
| 	else | ||||
| 	{ | ||||
| 		pxBlockToInsert->pxNextFreeBlock = pxIterator->pxNextFreeBlock;		 | ||||
| 	} | ||||
| 
 | ||||
| 	/* If the block being inserted plugged a gab, so was merged with the block
 | ||||
| 	before and the block after, then it's pxNextFreeBlock pointer will have | ||||
| 	already been set, and should not be set here as that would make it point | ||||
| 	to itself. */ | ||||
| 	if( pxIterator != pxBlockToInsert ) | ||||
| 	{ | ||||
| 		pxIterator->pxNextFreeBlock = pxBlockToInsert; | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
|  | @ -69,8 +69,10 @@ | |||
|  * which is in turn based on the ARM CM3 port. | ||||
|  *----------------------------------------------------------*/ | ||||
| 
 | ||||
| /* Scheduler includes. */ | ||||
| #include <xtensa/config/core.h> | ||||
| #include <malloc.h> | ||||
| #include <unistd.h> | ||||
| #include <stdio.h> | ||||
| 
 | ||||
| #include "FreeRTOS.h" | ||||
| #include "task.h" | ||||
|  | @ -79,6 +81,14 @@ | |||
| unsigned cpu_sr; | ||||
| char level1_int_disabled; | ||||
| 
 | ||||
| /* Supervisor stack pointer entry. This is the "high water mark" of how far the
 | ||||
|    supervisor stack grew down before task started. | ||||
| 
 | ||||
|    After tasks start, task stacks are all allocated from the heap and | ||||
|    FreeRTOS checks for stack overflow. | ||||
| */ | ||||
| static uint32_t xPortSupervisorStackPointer; | ||||
| 
 | ||||
| /*
 | ||||
|  * Stack initialization | ||||
|  */ | ||||
|  | @ -181,12 +191,37 @@ portBASE_TYPE xPortStartScheduler( void ) | |||
| 
 | ||||
|     vTaskSwitchContext(); | ||||
| 
 | ||||
|     /* mark the supervisor stack pointer high water mark. xt_int_exit
 | ||||
|        actually frees ~0x50 bytes off the stack, so this value is | ||||
|        conservative. | ||||
|     */ | ||||
|     __asm__ __volatile__ ("mov %0, a1\n" : "=a"(xPortSupervisorStackPointer)); | ||||
| 
 | ||||
|     sdk__xt_int_exit(); | ||||
| 
 | ||||
|     /* Should not get here as the tasks are now running! */ | ||||
|     return pdTRUE; | ||||
| } | ||||
| 
 | ||||
| /* Determine free heap size via libc sbrk function & mallinfo
 | ||||
| 
 | ||||
|    sbrk gives total size in totally unallocated memory, | ||||
|    mallinfo.fordblks gives free space inside area dedicated to heap. | ||||
| 
 | ||||
|    mallinfo is possibly non-portable, although glibc & newlib both support | ||||
|    the fordblks member. | ||||
| */ | ||||
| size_t xPortGetFreeHeapSize( void ) | ||||
| { | ||||
|     struct mallinfo mi = mallinfo(); | ||||
|     uint32_t brk_val = (uint32_t) sbrk(0); | ||||
| 
 | ||||
|     uint32_t sp = xPortSupervisorStackPointer; | ||||
|     if(sp == 0) /* scheduler not started */ | ||||
|         __asm__ __volatile__ ("mov %0, a1\n" : "=a"(sp)); | ||||
|     return sp - brk_val + mi.fordblks; | ||||
| } | ||||
| 
 | ||||
| void vPortEndScheduler( void ) | ||||
| { | ||||
|     /* No-op, nothing to return to */ | ||||
|  |  | |||
|  | @ -73,7 +73,6 @@ extern "C" { | |||
| #include "esp8266.h" | ||||
| #include "espressif/esp8266/ets_sys.h" | ||||
| #include <stdint.h> | ||||
| #include    <xtruntime.h> | ||||
| #include    "xtensa_rtos.h" | ||||
| #include "xtensa_interrupts.h" | ||||
| 
 | ||||
|  | @ -144,33 +143,31 @@ extern char sdk_NMIIrqIsOn; | |||
| extern char level1_int_disabled; | ||||
| extern unsigned cpu_sr; | ||||
| 
 | ||||
| /* ESPTODO: Currently we store the old interrupt level (ps) in a
 | ||||
|    global variable cpu_sr. It may not be necessary to do this, | ||||
|    especially as lx106 has only one real interrupt level + NMI, but it | ||||
|    all depends on how the blob libraries call into these functions. | ||||
| /* Disable interrupts, store old ps level in global variable cpu_sr.
 | ||||
| 
 | ||||
|    Note: cpu_sr is also referenced by the binary SDK. | ||||
| 
 | ||||
|    Where possible (and when writing non-FreeRTOS specific code), | ||||
|    prefer to _xt_disable_interrupts & _xt_enable_interrupts and store | ||||
|    the ps value in a local variable - that approach is recursive-safe | ||||
|    and generally better. | ||||
| */ | ||||
| inline static __attribute__((always_inline)) void _esp_disable_interrupts(void) | ||||
| inline static __attribute__((always_inline)) void portDISABLE_INTERRUPTS(void) | ||||
| { | ||||
|     if(!sdk_NMIIrqIsOn && !level1_int_disabled) { | ||||
| 	__asm__ volatile ("rsil %0, " XTSTR(XCHAL_EXCM_LEVEL) : "=a" (cpu_sr) :: "memory"); | ||||
| 	cpu_sr = _xt_disable_interrupts(); | ||||
| 	level1_int_disabled = 1; | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| inline static __attribute__((always_inline)) void _esp_enable_interrupts(void) | ||||
| inline static __attribute__((always_inline)) void portENABLE_INTERRUPTS(void) | ||||
| { | ||||
|     if(!sdk_NMIIrqIsOn && level1_int_disabled) { | ||||
| 	level1_int_disabled = 0; | ||||
| 	__asm__ volatile ("wsr %0, ps" :: "a" (cpu_sr) : "memory"); | ||||
|         _xt_restore_interrupts(cpu_sr); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| /* Disable interrupts, saving previous state in cpu_sr */ | ||||
| #define  portDISABLE_INTERRUPTS() _esp_disable_interrupts() | ||||
| 
 | ||||
| /* Restore interrupts to previous level saved in cpu_sr */ | ||||
| #define  portENABLE_INTERRUPTS() _esp_enable_interrupts() | ||||
| 
 | ||||
| /* Critical section management. */ | ||||
| void vPortEnterCritical( void ); | ||||
| void vPortExitCritical( void ); | ||||
|  |  | |||
|  | @ -1,77 +0,0 @@ | |||
| /*
 | ||||
|  * Stub functions called by binary espressif libraries | ||||
|  * | ||||
|  * Part of esp-open-rtos | ||||
|  * Copyright (C) 2015 Superhouse Automation Pty Ltd | ||||
|  * BSD Licensed as described in the file LICENSE | ||||
|  */ | ||||
| #include <stdlib.h> | ||||
| #include <string.h> | ||||
| #include "FreeRTOS.h" | ||||
| 
 | ||||
| /* SDK uses errno. errno was defined in libudhcp.a
 | ||||
|    but that library has been removed. */ | ||||
| int errno; | ||||
| 
 | ||||
| /* newlib uses __errno in some contexts */ | ||||
| int *__errno(void) { | ||||
|     return &errno; | ||||
| } | ||||
| 
 | ||||
| /* libc memory management functions.
 | ||||
| 
 | ||||
|    Many of these are linked from the RTOS SDK blob libraries. | ||||
| 
 | ||||
|    In esp_iot_rtos_sdk these are aliased to exception-safe versions of | ||||
|    the FreeRTOS functions. I think the rationale is that they're | ||||
|    sometimes called in exception contexts (ESPTODO: Verify this). | ||||
| 
 | ||||
|    For now these are exception-safe wrappers to the FreeRTOS versions. | ||||
|  */ | ||||
| 
 | ||||
| void *malloc(size_t nbytes) { | ||||
| 	void *res; | ||||
| 	portENTER_CRITICAL(); | ||||
| 	res = pvPortMalloc(nbytes); | ||||
| 	portEXIT_CRITICAL(); | ||||
| 	return res; | ||||
| } | ||||
| 
 | ||||
| void *calloc(size_t count, size_t size) { | ||||
| 	void *res; | ||||
| 	size_t nbytes = count * size; | ||||
| 	portENTER_CRITICAL(); | ||||
| 	res = pvPortMalloc(nbytes); | ||||
| 	portEXIT_CRITICAL(); | ||||
| 	if(res) { | ||||
| 		memset(res, 0, nbytes); | ||||
| 	} | ||||
| 	return res; | ||||
| } | ||||
| 
 | ||||
| void *zalloc(size_t nbytes) { | ||||
|     return calloc(1, nbytes); | ||||
| } | ||||
| 
 | ||||
| void *realloc(void *old, size_t newsize) { | ||||
| 	void *new; | ||||
| 	portENTER_CRITICAL(); | ||||
| 	if(newsize == 0) { | ||||
| 		vPortFree(old); | ||||
| 		return 0; | ||||
| 	} | ||||
| 	/* realloc implementation borrowed from esp_iot_rtos_sdk, could be better I think */ | ||||
| 	new = pvPortMalloc(newsize); | ||||
| 	if (new) { | ||||
| 		memcpy(new, old, newsize); | ||||
| 		vPortFree(old); | ||||
| 	} | ||||
| 	portEXIT_CRITICAL(); | ||||
| 	return new; | ||||
| } | ||||
| 
 | ||||
| void free(void *ptr) { | ||||
|     portENTER_CRITICAL(); | ||||
|     vPortFree(ptr); | ||||
|     portEXIT_CRITICAL(); | ||||
| } | ||||
|  | @ -5,5 +5,5 @@ INC_DIRS += $(freertos_MAIN)/include $(freertos_MAIN)/portable/esp8266 | |||
| freertos_ROOT = $(ROOT)FreeRTOS/ | ||||
| freertos_MAIN = $(freertos_ROOT)Source/ | ||||
| freertos_INC_DIR = $(freertos_MAIN)include $(freertos_MAIN)portable/esp8266 | ||||
| freertos_SRC_DIR = $(freertos_MAIN) $(freertos_MAIN)portable/MemMang $(freertos_MAIN)portable/esp8266 | ||||
| freertos_SRC_DIR = $(freertos_MAIN) $(freertos_MAIN)portable/esp8266 | ||||
| $(eval $(call component_compile_rules,freertos)) | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue