mirror of
				https://github.com/pvvx/RTL00MP3.git
				synced 2025-07-31 12:41:06 +00:00 
			
		
		
		
	first commit
This commit is contained in:
		
							parent
							
								
									2ee525362e
								
							
						
					
					
						commit
						d108756e9b
					
				
					 792 changed files with 336059 additions and 0 deletions
				
			
		
							
								
								
									
										263
									
								
								project/src/driver/i2s_freertos.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										263
									
								
								project/src/driver/i2s_freertos.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,263 @@ | |||
| /******************************************************************************
 | ||||
|  * | ||||
|  * FileName: i2s_freertos.c | ||||
|  * | ||||
|  * Description: I2S output routines for a FreeRTOS system. | ||||
|  * | ||||
|  * Modification history: | ||||
|  *     2015/10, RTL8710 kissste, pvvx | ||||
| *******************************************************************************/ | ||||
| 
 | ||||
| /*
 | ||||
| How does this work? Basically, to get sound, you need to: | ||||
| - Connect an I2S codec to the I2S pins on the RTL. | ||||
| - Start up a thread that's going to do the sound output | ||||
| - Call I2sInit() | ||||
| - Call I2sSetRate() with the sample rate you want. | ||||
| - Generate sound and call i2sPushSample() with 32-bit samples. | ||||
| The 32bit samples basically are 2 16-bit signed values (the analog values for | ||||
| the left and right channel) concatenated as (Rout<<16)+Lout | ||||
| 
 | ||||
| I2sPushSample will block when you're sending data too quickly, so you can just | ||||
| generate and push data as fast as you can and I2sPushSample will regulate the | ||||
| speed. | ||||
| */ | ||||
| 
 | ||||
| #include "FreeRTOS.h" | ||||
| #include "task.h" | ||||
| #include "semphr.h" | ||||
| #include "queue.h" | ||||
| #include "user/playerconfig.h" | ||||
| 
 | ||||
| #include "i2s_api.h" | ||||
| #include "driver/i2s_freertos.h" | ||||
| 
 | ||||
| #define USE_RTL_I2S_API  0 // speed
 | ||||
| 
 | ||||
| PI2S_OBJS pi2s[MAX_I2S_OBJS]; // I2S0, I2S1
 | ||||
| 
 | ||||
| // i2s interrupt callback
 | ||||
| static void i2s_test_tx_complete(void *data, char *pbuf) | ||||
| { | ||||
| #if I2S_DEBUG_LEVEL > 1 | ||||
|     i2s_t *i2s_obj = (i2s_t *)data; | ||||
| 	int idx = i2s_obj->InitDat.I2SIdx; | ||||
| 	int reg = HAL_I2S_READ32(idx, REG_I2S_TX_PAGE0_OWN); | ||||
| 	reg |= HAL_I2S_READ32(idx, REG_I2S_TX_PAGE1_OWN); | ||||
| 	reg |= HAL_I2S_READ32(idx, REG_I2S_TX_PAGE2_OWN); | ||||
| 	reg |= HAL_I2S_READ32(idx, REG_I2S_TX_PAGE3_OWN); | ||||
| 	if(!(reg & BIT_PAGE_I2S_OWN_BIT)) pi2s[idx]->underrunCnt++; | ||||
| #endif | ||||
| } | ||||
| 
 | ||||
| void i2sClose(int mask) { | ||||
| 	int i; | ||||
| 	for(i = 0; i < MAX_I2S_OBJS; i++) { | ||||
| 		if(mask & (1 << i)) { | ||||
| 			if(pi2s[i] != NULL)	{ | ||||
| 				if(pi2s[i]->i2s_obj.InitDat.I2SEn != I2S_DISABLE) { | ||||
| 					i2s_disable(&pi2s[i]->i2s_obj); // HalI2SDisable(&pi2s[i]->i2s_obj.I2SAdapter);
 | ||||
| 					i2s_deinit(&pi2s[i]->i2s_obj); // HalI2SDeInit(&pi2s[i]->i2s_obj.I2SAdapter);
 | ||||
| #if I2S_DEBUG_LEVEL > 0 | ||||
| 					DBG_8195A("I2S%d: i2s_disable (%d)\n", i, pi2s[i]->i2s_obj.InitDat.I2SEn); | ||||
| #endif | ||||
| 				} | ||||
| 				if(pi2s[i]->i2s_obj.InitDat.I2STxData != NULL) { | ||||
| 					vPortFree(pi2s[i]->i2s_obj.InitDat.I2STxData); | ||||
| 					pi2s[i]->i2s_obj.InitDat.I2STxData = NULL; | ||||
| 				} | ||||
| 				vPortFree(pi2s[i]); | ||||
| 				pi2s[i] = NULL; | ||||
| 				if(i==0) HalPinCtrlRtl8195A(JTAG, 0, 1); | ||||
| 				DBG_8195A("I2S%d: Closed.\n", i); | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| //Initialize I2S subsystem for DMA circular buffer use
 | ||||
| int i2sInit(int mask, int bufsize, int word_len) { // word_len = WL_16b or WL_24b
 | ||||
| #if I2S_DEBUG_LEVEL > 2 | ||||
| 	DBG_ERR_MSG_ON(_DBG_I2S_ | _DBG_GDMA_); | ||||
| 	DBG_INFO_MSG_ON(_DBG_I2S_ | _DBG_GDMA_); | ||||
| 	DBG_WARN_MSG_ON(_DBG_I2S_ | _DBG_GDMA_); | ||||
| #endif | ||||
| 	if(bufsize < I2S_DMA_PAGE_SIZE_MS_96K*2) { | ||||
| 		DBG_8195A("I2S: Min buffer %d bytes!\n", I2S_DMA_PAGE_SIZE_MS_96K*2); | ||||
| 		return 0; | ||||
| 	} | ||||
| 	int page_size = bufsize * sizeof(u32); | ||||
| 	if(word_len != WL_16b) page_size <<= 1; //bufsize *2;
 | ||||
| 	int i; | ||||
| 	for(i = 0; i < MAX_I2S_OBJS; i++) { | ||||
| 		if (mask & (1 << i)) { | ||||
| 			if(pi2s[i] != NULL) i2sClose(1 << i); | ||||
| 			PI2S_OBJS pi2s_new = pvPortMalloc(sizeof(I2S_OBJS)); | ||||
| 			if(pi2s_new == NULL) { | ||||
| 		        DBG_8195A("I2S%d: Not heap buffer %d bytes!\n", i, sizeof(i2s_t) + page_size * I2S_DMA_PAGE_NUM); | ||||
| 				return 0; | ||||
| 			} | ||||
| 			rtl_memset(pi2s_new, 0, sizeof(i2s_t)); | ||||
| 			u8 * i2s_tx_buf = (u8 *) pvPortMalloc(page_size * I2S_DMA_PAGE_NUM); | ||||
| 		    if (i2s_tx_buf == NULL) { | ||||
| 		    	vPortFree(pi2s_new); | ||||
| 		        DBG_8195A("I2S%d: Not heap buffer %d bytes!\n", i, sizeof(i2s_t) + page_size * I2S_DMA_PAGE_NUM); | ||||
| 		    	return 0; | ||||
| 		    } | ||||
| 		    pi2s[i] = pi2s_new; | ||||
| #if I2S_DEBUG_LEVEL > 1 | ||||
| 		    pi2s_new->underrunCnt = 0; | ||||
| #endif | ||||
| 		    pi2s[i]->sampl_err = 0; | ||||
| 		    pi2s_new->currDMABuffPos = 0; | ||||
| 		    pi2s_new->currDMABuff = NULL; | ||||
| 
 | ||||
| 		    i2s_t * pi2s_obj = &pi2s_new->i2s_obj; | ||||
| 
 | ||||
| 		    pi2s_obj->channel_num = CH_STEREO; | ||||
| 		    pi2s_obj->sampling_rate = SR_96KHZ; | ||||
| 		    pi2s_obj->word_length = word_len; | ||||
| 		    pi2s_obj->direction = I2S_DIR_TX; //consider switching to TX only
 | ||||
| 			if(i == 0) { | ||||
| 				HalPinCtrlRtl8195A(JTAG, 0, 0); | ||||
| 				i2s_init(pi2s_obj, I2S0_SCLK_PIN, I2S0_WS_PIN, I2S0_SD_PIN); | ||||
| 			} | ||||
| 			else i2s_init(pi2s_obj, I2S1_SCLK_PIN, I2S1_WS_PIN, I2S1_SD_PIN); | ||||
| 			i2s_set_param(pi2s_obj, pi2s_obj->channel_num, pi2s_obj->sampling_rate, pi2s_obj->word_length); | ||||
| 		    i2s_set_dma_buffer(pi2s_obj, i2s_tx_buf, NULL, I2S_DMA_PAGE_NUM, page_size); | ||||
| 		    i2s_tx_irq_handler(pi2s_obj, i2s_test_tx_complete, (uint32_t)pi2s_obj); | ||||
| 	//    	i2s_rx_irq_handler(pi2s_obj, (i == 0)? (i2s_irq_handler)i2s1_test_rx_complete : (i2s_irq_handler)i2s2_test_rx_complete, i); // TX only!
 | ||||
| 		    i2s_enable(pi2s_obj); | ||||
| 		    DBG_8195A("I2S%d: Alloc DMA buf %d bytes (%d x %d samples %d bits)\n", i, page_size * I2S_DMA_PAGE_NUM, I2S_DMA_PAGE_NUM, bufsize, (word_len == WL_16b)? 32 : 96); | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| //Set the I2S sample rate, in HZ
 | ||||
| char i2sSetRate(int mask, int rate) { | ||||
| 
 | ||||
| 	int sample_rate; | ||||
| 	char result = 1; | ||||
| #if defined(OVERSAMPLES) && defined(PWM_HACK96BIT) | ||||
| 	rate <<= 1; | ||||
| 	while (rate <= 48000) { | ||||
| 		rate <<= 1; | ||||
| 		result++; | ||||
| 	} | ||||
| #endif | ||||
| 	if (rate>=96000) sample_rate = SR_96KHZ; | ||||
| 	else if (rate>=88200) sample_rate = SR_88p2KHZ; | ||||
|     else if (rate>=48000) sample_rate = SR_48KHZ; | ||||
|     else if (rate>=44100) sample_rate = SR_44p1KHZ; | ||||
|     else if (rate>=32000) sample_rate = SR_32KHZ; | ||||
| 	else if (rate>=24000) sample_rate = SR_24KHZ; | ||||
| 	else if (rate>=22050) sample_rate = SR_22p05KHZ; | ||||
| 	else if (rate>=16000) sample_rate = SR_16KHZ; | ||||
| 	else if (rate>=11020) sample_rate = SR_11p02KHZ; | ||||
| 	else if (rate>= 8000) sample_rate = SR_8KHZ; | ||||
| 	else sample_rate = SR_7p35KHZ; | ||||
| 	int i; | ||||
| 	for(i = 0; i < MAX_I2S_OBJS; i++) { | ||||
| 		if (mask & (1 << i)) { | ||||
| 			i2s_t * pi2s_obj = &pi2s[i]->i2s_obj; | ||||
| 			pi2s[i]->sampl_err = 0; | ||||
| 			pi2s_obj->sampling_rate = sample_rate; | ||||
| #if USE_RTL_I2S_API | ||||
| 			i2s_set_param(pi2s_obj, pi2s_obj->channel_num, pi2s_obj->sampling_rate, pi2s_obj->word_length); | ||||
| #else | ||||
| 			pi2s_obj->I2SAdapter.pInitDat->I2SRate = sample_rate; | ||||
| 			HalI2SSetRate(pi2s_obj->I2SAdapter.pInitDat); | ||||
| #endif | ||||
| 		} | ||||
| 	} | ||||
| 	DBG_8195A("I2S: Set Sample Rate %d (x%d)\n", rate, result); | ||||
| 	return result; | ||||
| } | ||||
| 
 | ||||
| #if defined(PWM_HACK96BIT) | ||||
| //This routine pushes a single, 32-bit sample to the I2S buffers. Call this at (on average) 
 | ||||
| //at least the current sample rate. You can also call it quicker: it will suspend the calling
 | ||||
| //thread if the buffer is full and resume when there's room again.
 | ||||
| u32 i2sPushPWMSamples(u32 sample) { | ||||
| 	for(int i = 0; i < MAX_I2S_OBJS; i++) { | ||||
| 		PI2S_OBJS pi2s_cur = pi2s[i]; | ||||
| 		PHAL_I2S_ADAPTER I2SAdapter = &pi2s_cur->i2s_obj.I2SAdapter; | ||||
| 		while(pi2s_cur->currDMABuff == NULL){ | ||||
| #if USE_RTL_I2S_API | ||||
| 			pi2s_cur->currDMABuff = i2s_get_tx_page(&pi2s_cur->i2s_obj); | ||||
| 			if(pi2s_cur->currDMABuff == NULL) vTaskDelay(I2S_DMA_PAGE_WAIT_MS_MIN); | ||||
| #else | ||||
| 			u8 page_idx = HalI2SGetTxPage((VOID*)I2SAdapter->pInitDat); | ||||
| 			if(page_idx < I2S_DMA_PAGE_NUM)	pi2s_cur->currDMABuff = ((u32 *)I2SAdapter->TxPageList[page_idx]); | ||||
| 			else vTaskDelay(I2S_DMA_PAGE_WAIT_MS_MIN); | ||||
| #endif | ||||
| 			pi2s_cur->currDMABuffPos = 0; | ||||
| 		} | ||||
| 		u32 *p = &pi2s_cur->currDMABuff[pi2s_cur->currDMABuffPos]; | ||||
| 		if(i) sample >>= 16; | ||||
| 		s32 smp = (s16)sample + 0x8000 + pi2s_cur->sampl_err; | ||||
| 		if (smp > 0xffff) smp = 0xffff; | ||||
| 		else if (smp < 0) smp = 0; | ||||
| 		u8 x = smp/(u16)(0x10000/97); | ||||
| 		pi2s_cur->sampl_err = smp - x * (u16)(0x10000/97); | ||||
| 		if(x < 24) { | ||||
| 			*p++ = (1 << x) -1; | ||||
| 			*p++ = 0; | ||||
| 			*p++ = 0; | ||||
| 			*p = 0; | ||||
| 		} | ||||
| 		else if (x < 48) { | ||||
| 			*p++ = 0xFFFFFFFF; | ||||
| 			*p++ = (1 << (x - 24)) -1; | ||||
| 			*p++ = 0; | ||||
| 			*p = 0; | ||||
| 		} | ||||
| 		else if (x < 72) { | ||||
| 			*p++ = 0xFFFFFFFF; | ||||
| 			*p++ = 0xFFFFFFFF; | ||||
| 			*p++ = (1 << (x - 48)) -1; | ||||
| 			*p = 0; | ||||
| 		} | ||||
| 		else if (x < 96) { | ||||
| 			*p++ = 0xFFFFFFFF; | ||||
| 			*p++ = 0xFFFFFFFF; | ||||
| 			*p++ = 0xFFFFFFFF; | ||||
| 			*p = (1 << (x - 72)) -1; | ||||
| 		} | ||||
| 		else { | ||||
| 			*p++ = 0xFFFFFFFF; | ||||
| 			*p++ = 0xFFFFFFFF; | ||||
| 			*p++ = 0xFFFFFFFF; | ||||
| 			*p = 0xFFFFFFFF; | ||||
| 		} | ||||
| 		pi2s_cur->currDMABuffPos += 4; | ||||
| 	} | ||||
| 	portENTER_CRITICAL(); | ||||
| 	for(int i = 0; i < MAX_I2S_OBJS; i++) { | ||||
| 		PI2S_OBJS pi2s_cur = pi2s[i]; | ||||
| 		if (pi2s_cur->currDMABuffPos > pi2s_cur->i2s_obj.InitDat.I2SPageSize) { | ||||
| #if USE_RTL_I2S_API | ||||
| 			i2s_send_page(&pi2s_cur->i2s_obj, pi2s_cur->currDMABuff); | ||||
| #else | ||||
| 			PHAL_I2S_ADAPTER I2SAdapter = &pi2s_cur->i2s_obj.I2SAdapter; | ||||
| 			int n; | ||||
| 			for (n = 0; n < I2S_DMA_PAGE_NUM; n++) { | ||||
| 				if (I2SAdapter->TxPageList[n] == pi2s_cur->currDMABuff) { | ||||
| 					HalI2SPageSend(I2SAdapter->pInitDat, n); | ||||
| 					HAL_I2S_WRITE32(i, REG_I2S_TX_PAGE0_OWN + 4 * n, BIT_PAGE_I2S_OWN_BIT); | ||||
| 					break;  // break the for loop
 | ||||
| 				} | ||||
| 			} | ||||
| #endif | ||||
| 			pi2s_cur->currDMABuff = NULL; | ||||
| 		} | ||||
| 	} | ||||
| 	portEXIT_CRITICAL(); | ||||
| } | ||||
| #endif | ||||
| 
 | ||||
| #if I2S_DEBUG_LEVEL > 1 | ||||
| long i2s1GetUnderrunCnt(int num) { | ||||
| 	return pi2s[num]->underrunCnt; | ||||
| } | ||||
| #endif | ||||
							
								
								
									
										14
									
								
								project/src/mad/align.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								project/src/mad/align.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,14 @@ | |||
| 
 | ||||
| #include "rtl8195a/rtl_common.h" | ||||
| /*
 | ||||
| char unalChar(const char *adr) { | ||||
| 	return (*((unsigned int *)((unsigned int)adr & (~3))))>>(((unsigned int)adr & 3) << 3); | ||||
| } | ||||
| */ | ||||
| 
 | ||||
| short unalShort(const short *adr) { | ||||
| 	int *p=(int *)((int)adr&(~3)); | ||||
| 	int v=*p; | ||||
| 	int w=((int)adr&3); | ||||
| 	if (w==0) return v; else return (v>>16); | ||||
| } | ||||
							
								
								
									
										236
									
								
								project/src/mad/bit.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										236
									
								
								project/src/mad/bit.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,236 @@ | |||
| /*
 | ||||
|  * libmad - MPEG audio decoder library | ||||
|  * Copyright (C) 2000-2004 Underbit Technologies, Inc. | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|  * the Free Software Foundation; either version 2 of the License, or | ||||
|  * (at your option) any later version. | ||||
|  * | ||||
|  * This program 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.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  * | ||||
|  * You should have received a copy of the GNU General Public License | ||||
|  * along with this program; if not, write to the Free Software | ||||
|  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA | ||||
|  * | ||||
|  * $Id: bit.c,v 1.12 2004/01/23 09:41:32 rob Exp $ | ||||
|  */ | ||||
| 
 | ||||
| # ifdef HAVE_CONFIG_H | ||||
| #  include "config.h" | ||||
| # endif | ||||
| 
 | ||||
| # include "global.h" | ||||
| 
 | ||||
| # ifdef HAVE_LIMITS_H | ||||
| #  include <limits.h> | ||||
| # else | ||||
| #  define CHAR_BIT  8 | ||||
| # endif | ||||
| 
 | ||||
| # include "bit.h" | ||||
| 
 | ||||
| /*
 | ||||
|  * This is the lookup table for computing the CRC-check word. | ||||
|  * As described in section 2.4.3.1 and depicted in Figure A.9 | ||||
|  * of ISO/IEC 11172-3, the generator polynomial is: | ||||
|  * | ||||
|  * G(X) = X^16 + X^15 + X^2 + 1 | ||||
|  */ | ||||
| static | ||||
| unsigned short const ICACHE_RODATA_ATTR crc_table[256] = { | ||||
|   0x0000, 0x8005, 0x800f, 0x000a, 0x801b, 0x001e, 0x0014, 0x8011, | ||||
|   0x8033, 0x0036, 0x003c, 0x8039, 0x0028, 0x802d, 0x8027, 0x0022, | ||||
|   0x8063, 0x0066, 0x006c, 0x8069, 0x0078, 0x807d, 0x8077, 0x0072, | ||||
|   0x0050, 0x8055, 0x805f, 0x005a, 0x804b, 0x004e, 0x0044, 0x8041, | ||||
|   0x80c3, 0x00c6, 0x00cc, 0x80c9, 0x00d8, 0x80dd, 0x80d7, 0x00d2, | ||||
|   0x00f0, 0x80f5, 0x80ff, 0x00fa, 0x80eb, 0x00ee, 0x00e4, 0x80e1, | ||||
|   0x00a0, 0x80a5, 0x80af, 0x00aa, 0x80bb, 0x00be, 0x00b4, 0x80b1, | ||||
|   0x8093, 0x0096, 0x009c, 0x8099, 0x0088, 0x808d, 0x8087, 0x0082, | ||||
| 
 | ||||
|   0x8183, 0x0186, 0x018c, 0x8189, 0x0198, 0x819d, 0x8197, 0x0192, | ||||
|   0x01b0, 0x81b5, 0x81bf, 0x01ba, 0x81ab, 0x01ae, 0x01a4, 0x81a1, | ||||
|   0x01e0, 0x81e5, 0x81ef, 0x01ea, 0x81fb, 0x01fe, 0x01f4, 0x81f1, | ||||
|   0x81d3, 0x01d6, 0x01dc, 0x81d9, 0x01c8, 0x81cd, 0x81c7, 0x01c2, | ||||
|   0x0140, 0x8145, 0x814f, 0x014a, 0x815b, 0x015e, 0x0154, 0x8151, | ||||
|   0x8173, 0x0176, 0x017c, 0x8179, 0x0168, 0x816d, 0x8167, 0x0162, | ||||
|   0x8123, 0x0126, 0x012c, 0x8129, 0x0138, 0x813d, 0x8137, 0x0132, | ||||
|   0x0110, 0x8115, 0x811f, 0x011a, 0x810b, 0x010e, 0x0104, 0x8101, | ||||
| 
 | ||||
|   0x8303, 0x0306, 0x030c, 0x8309, 0x0318, 0x831d, 0x8317, 0x0312, | ||||
|   0x0330, 0x8335, 0x833f, 0x033a, 0x832b, 0x032e, 0x0324, 0x8321, | ||||
|   0x0360, 0x8365, 0x836f, 0x036a, 0x837b, 0x037e, 0x0374, 0x8371, | ||||
|   0x8353, 0x0356, 0x035c, 0x8359, 0x0348, 0x834d, 0x8347, 0x0342, | ||||
|   0x03c0, 0x83c5, 0x83cf, 0x03ca, 0x83db, 0x03de, 0x03d4, 0x83d1, | ||||
|   0x83f3, 0x03f6, 0x03fc, 0x83f9, 0x03e8, 0x83ed, 0x83e7, 0x03e2, | ||||
|   0x83a3, 0x03a6, 0x03ac, 0x83a9, 0x03b8, 0x83bd, 0x83b7, 0x03b2, | ||||
|   0x0390, 0x8395, 0x839f, 0x039a, 0x838b, 0x038e, 0x0384, 0x8381, | ||||
| 
 | ||||
|   0x0280, 0x8285, 0x828f, 0x028a, 0x829b, 0x029e, 0x0294, 0x8291, | ||||
|   0x82b3, 0x02b6, 0x02bc, 0x82b9, 0x02a8, 0x82ad, 0x82a7, 0x02a2, | ||||
|   0x82e3, 0x02e6, 0x02ec, 0x82e9, 0x02f8, 0x82fd, 0x82f7, 0x02f2, | ||||
|   0x02d0, 0x82d5, 0x82df, 0x02da, 0x82cb, 0x02ce, 0x02c4, 0x82c1, | ||||
|   0x8243, 0x0246, 0x024c, 0x8249, 0x0258, 0x825d, 0x8257, 0x0252, | ||||
|   0x0270, 0x8275, 0x827f, 0x027a, 0x826b, 0x026e, 0x0264, 0x8261, | ||||
|   0x0220, 0x8225, 0x822f, 0x022a, 0x823b, 0x023e, 0x0234, 0x8231, | ||||
|   0x8213, 0x0216, 0x021c, 0x8219, 0x0208, 0x820d, 0x8207, 0x0202 | ||||
| }; | ||||
| 
 | ||||
| # define CRC_POLY  0x8005 | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	bit->init() | ||||
|  * DESCRIPTION:	initialize bit pointer struct | ||||
|  */ | ||||
| void mad_bit_init(struct mad_bitptr *bitptr, unsigned char const *byte) | ||||
| { | ||||
|   bitptr->byte  = byte; | ||||
|   bitptr->cache = 0; | ||||
|   bitptr->left  = CHAR_BIT; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	bit->length() | ||||
|  * DESCRIPTION:	return number of bits between start and end points | ||||
|  */ | ||||
| unsigned int mad_bit_length(struct mad_bitptr const *begin, | ||||
| 			    struct mad_bitptr const *end) | ||||
| { | ||||
|   return begin->left + | ||||
|     CHAR_BIT * (end->byte - (begin->byte + 1)) + (CHAR_BIT - end->left); | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	bit->nextbyte() | ||||
|  * DESCRIPTION:	return pointer to next unprocessed byte | ||||
|  */ | ||||
| unsigned char const *mad_bit_nextbyte(struct mad_bitptr const *bitptr) | ||||
| { | ||||
|   return bitptr->left == CHAR_BIT ? bitptr->byte : bitptr->byte + 1; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	bit->skip() | ||||
|  * DESCRIPTION:	advance bit pointer | ||||
|  */ | ||||
| void mad_bit_skip(struct mad_bitptr *bitptr, unsigned int len) | ||||
| { | ||||
|   bitptr->byte += len / CHAR_BIT; | ||||
|   bitptr->left -= len % CHAR_BIT; | ||||
| 
 | ||||
|   if (bitptr->left > CHAR_BIT) { | ||||
|     bitptr->byte++; | ||||
|     bitptr->left += CHAR_BIT; | ||||
|   } | ||||
| 
 | ||||
|   if (bitptr->left < CHAR_BIT) | ||||
|     bitptr->cache = *bitptr->byte; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	bit->read() | ||||
|  * DESCRIPTION:	read an arbitrary number of bits and return their UIMSBF value | ||||
|  */ | ||||
| unsigned long mad_bit_read(struct mad_bitptr *bitptr, unsigned int len) | ||||
| { | ||||
|   register unsigned long value; | ||||
| 
 | ||||
|   if (bitptr->left == CHAR_BIT) | ||||
|     bitptr->cache = *bitptr->byte; | ||||
| 
 | ||||
|   if (len < bitptr->left) { | ||||
|     value = (bitptr->cache & ((1 << bitptr->left) - 1)) >> | ||||
|       (bitptr->left - len); | ||||
|     bitptr->left -= len; | ||||
| 
 | ||||
|     return value; | ||||
|   } | ||||
| 
 | ||||
|   /* remaining bits in current byte */ | ||||
| 
 | ||||
|   value = bitptr->cache & ((1 << bitptr->left) - 1); | ||||
|   len  -= bitptr->left; | ||||
| 
 | ||||
|   bitptr->byte++; | ||||
|   bitptr->left = CHAR_BIT; | ||||
| 
 | ||||
|   /* more bytes */ | ||||
| 
 | ||||
|   while (len >= CHAR_BIT) { | ||||
|     value = (value << CHAR_BIT) | *bitptr->byte++; | ||||
|     len  -= CHAR_BIT; | ||||
|   } | ||||
| 
 | ||||
|   if (len > 0) { | ||||
|     bitptr->cache = *bitptr->byte; | ||||
| 
 | ||||
|     value = (value << len) | (bitptr->cache >> (CHAR_BIT - len)); | ||||
|     bitptr->left -= len; | ||||
|   } | ||||
| 
 | ||||
|   return value; | ||||
| } | ||||
| 
 | ||||
| # if 0 | ||||
| /*
 | ||||
|  * NAME:	bit->write() | ||||
|  * DESCRIPTION:	write an arbitrary number of bits | ||||
|  */ | ||||
| void mad_bit_write(struct mad_bitptr *bitptr, unsigned int len, | ||||
| 		   unsigned long value) | ||||
| { | ||||
|   unsigned char *ptr; | ||||
| 
 | ||||
|   ptr = (unsigned char *) bitptr->byte; | ||||
| 
 | ||||
|   /* ... */ | ||||
| } | ||||
| # endif | ||||
| 
 | ||||
| 
 | ||||
| //extern short unalShort(const short *adr);
 | ||||
| /*
 | ||||
|  * NAME:	bit->crc() | ||||
|  * DESCRIPTION:	compute CRC-check word | ||||
|  */ | ||||
| unsigned short mad_bit_crc(struct mad_bitptr bitptr, unsigned int len, | ||||
| 			   unsigned short init) | ||||
| { | ||||
|   register unsigned int crc; | ||||
| 
 | ||||
|   for (crc = init; len >= 32; len -= 32) { | ||||
|     register unsigned long data; | ||||
| 
 | ||||
|     data = mad_bit_read(&bitptr, 32); | ||||
| 
 | ||||
|     crc = (crc << 8) ^ crc_table[((crc >> 8) ^ (data >> 24)) & 0xff]; | ||||
|     crc = (crc << 8) ^ crc_table[((crc >> 8) ^ (data >> 16)) & 0xff]; | ||||
|     crc = (crc << 8) ^ crc_table[((crc >> 8) ^ (data >>  8)) & 0xff]; | ||||
|     crc = (crc << 8) ^ crc_table[((crc >> 8) ^ (data >>  0)) & 0xff]; | ||||
|   } | ||||
| 
 | ||||
|   switch (len / 8) { | ||||
|   case 3: crc = (crc << 8) ^ crc_table[((crc >> 8) ^ mad_bit_read(&bitptr, 8)) & 0xff]; | ||||
|   case 2: crc = (crc << 8) ^ crc_table[((crc >> 8) ^ mad_bit_read(&bitptr, 8)) & 0xff]; | ||||
|   case 1: crc = (crc << 8) ^ crc_table[((crc >> 8) ^ mad_bit_read(&bitptr, 8)) & 0xff]; | ||||
|    | ||||
|   len %= 8; | ||||
| 
 | ||||
|   case 0: break; | ||||
|   } | ||||
| 
 | ||||
|   while (len--) { | ||||
|     register unsigned int msb; | ||||
| 
 | ||||
|     msb = mad_bit_read(&bitptr, 1) ^ (crc >> 15); | ||||
| 
 | ||||
|     crc <<= 1; | ||||
|     if (msb & 1) | ||||
|       crc ^= CRC_POLY; | ||||
|   } | ||||
| 
 | ||||
|   return crc & 0xffff; | ||||
| } | ||||
							
								
								
									
										577
									
								
								project/src/mad/decoder.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										577
									
								
								project/src/mad/decoder.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,577 @@ | |||
| /*
 | ||||
|  * libmad - MPEG audio decoder library | ||||
|  * Copyright (C) 2000-2004 Underbit Technologies, Inc. | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|  * the Free Software Foundation; either version 2 of the License, or | ||||
|  * (at your option) any later version. | ||||
|  * | ||||
|  * This program 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.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  * | ||||
|  * You should have received a copy of the GNU General Public License | ||||
|  * along with this program; if not, write to the Free Software | ||||
|  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA | ||||
|  * | ||||
|  * $Id: decoder.c,v 1.22 2004/01/23 09:41:32 rob Exp $ | ||||
|  */ | ||||
| 
 | ||||
| # ifdef HAVE_CONFIG_H | ||||
| #  include "config.h" | ||||
| # endif | ||||
| 
 | ||||
| # include "global.h" | ||||
| 
 | ||||
| # ifdef HAVE_SYS_TYPES_H | ||||
| #  include <sys/types.h> | ||||
| # endif | ||||
| 
 | ||||
| # ifdef HAVE_SYS_WAIT_H | ||||
| #  include <sys/wait.h> | ||||
| # endif | ||||
| 
 | ||||
| # ifdef HAVE_UNISTD_H | ||||
| #  include <unistd.h> | ||||
| # endif | ||||
| 
 | ||||
| # ifdef HAVE_FCNTL_H | ||||
| #  include <fcntl.h> | ||||
| # endif | ||||
| 
 | ||||
| # include <stdlib.h> | ||||
| 
 | ||||
| # ifdef HAVE_ERRNO_H | ||||
| #  include <errno.h> | ||||
| # endif | ||||
| 
 | ||||
| # include "stream.h" | ||||
| # include "frame.h" | ||||
| # include "synth.h" | ||||
| # include "decoder.h" | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	decoder->init() | ||||
|  * DESCRIPTION:	initialize a decoder object with callback routines | ||||
|  */ | ||||
| void ICACHE_FLASH_ATTR mad_decoder_init(struct mad_decoder *decoder, void *data, | ||||
| 		enum mad_flow (*input_func)(void *, struct mad_stream *), | ||||
| 		enum mad_flow (*header_func)(void *, struct mad_header const *), | ||||
| 		enum mad_flow (*filter_func)(void *, struct mad_stream const *, | ||||
| 				struct mad_frame *), | ||||
| 		enum mad_flow (*output_func)(void *, struct mad_header const *, | ||||
| 				struct mad_pcm *), | ||||
| 		enum mad_flow (*error_func)(void *, struct mad_stream *, | ||||
| 				struct mad_frame *), | ||||
| 		enum mad_flow (*message_func)(void *, void *, unsigned int *)) { | ||||
| 	decoder->mode = -1; | ||||
| 
 | ||||
| 	decoder->options = 0; | ||||
| 
 | ||||
| 	decoder->async.pid = 0; | ||||
| 	decoder->async.in = -1; | ||||
| 	decoder->async.out = -1; | ||||
| 
 | ||||
| 	decoder->sync = 0; | ||||
| 
 | ||||
| 	decoder->cb_data = data; | ||||
| 
 | ||||
| 	decoder->input_func = input_func; | ||||
| 	decoder->header_func = header_func; | ||||
| 	decoder->filter_func = filter_func; | ||||
| 	decoder->output_func = output_func; | ||||
| 	decoder->error_func = error_func; | ||||
| 	decoder->message_func = message_func; | ||||
| } | ||||
| 
 | ||||
| int ICACHE_FLASH_ATTR mad_decoder_finish(struct mad_decoder *decoder) { | ||||
| # if defined(USE_ASYNC) | ||||
| 	if (decoder->mode == MAD_DECODER_MODE_ASYNC && decoder->async.pid) { | ||||
| 		pid_t pid; | ||||
| 		int status; | ||||
| 
 | ||||
| 		close(decoder->async.in); | ||||
| 
 | ||||
| 		do | ||||
| 		pid = waitpid(decoder->async.pid, &status, 0); | ||||
| 		while (pid == -1 && errno == EINTR); | ||||
| 
 | ||||
| 		decoder->mode = -1; | ||||
| 
 | ||||
| 		close(decoder->async.out); | ||||
| 
 | ||||
| 		decoder->async.pid = 0; | ||||
| 		decoder->async.in = -1; | ||||
| 		decoder->async.out = -1; | ||||
| 
 | ||||
| 		if (pid == -1) | ||||
| 		return -1; | ||||
| 
 | ||||
| 		return (!WIFEXITED(status) || WEXITSTATUS(status)) ? -1 : 0; | ||||
| 	} | ||||
| # endif | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| # if defined(USE_ASYNC) | ||||
| static | ||||
| enum mad_flow ICACHE_FLASH_ATTR send_io(int fd, void const *data, size_t len) | ||||
| { | ||||
| 	char const *ptr = data; | ||||
| 	ssize_t count; | ||||
| 
 | ||||
| 	while (len) { | ||||
| 		do | ||||
| 		count = write(fd, ptr, len); | ||||
| 		while (count == -1 && errno == EINTR); | ||||
| 
 | ||||
| 		if (count == -1) | ||||
| 		return MAD_FLOW_BREAK; | ||||
| 
 | ||||
| 		len -= count; | ||||
| 		ptr += count; | ||||
| 	} | ||||
| 
 | ||||
| 	return MAD_FLOW_CONTINUE; | ||||
| } | ||||
| 
 | ||||
| static | ||||
| enum mad_flow ICACHE_FLASH_ATTR receive_io(int fd, void *buffer, size_t len) | ||||
| { | ||||
| 	char *ptr = buffer; | ||||
| 	ssize_t count; | ||||
| 
 | ||||
| 	while (len) { | ||||
| 		do | ||||
| 		count = read(fd, ptr, len); | ||||
| 		while (count == -1 && errno == EINTR); | ||||
| 
 | ||||
| 		if (count == -1) | ||||
| 		return (errno == EAGAIN) ? MAD_FLOW_IGNORE : MAD_FLOW_BREAK; | ||||
| 		else if (count == 0) | ||||
| 		return MAD_FLOW_STOP; | ||||
| 
 | ||||
| 		len -= count; | ||||
| 		ptr += count; | ||||
| 	} | ||||
| 
 | ||||
| 	return MAD_FLOW_CONTINUE; | ||||
| } | ||||
| 
 | ||||
| static | ||||
| enum mad_flow ICACHE_FLASH_ATTR receive_io_blocking(int fd, void *buffer, size_t len) | ||||
| { | ||||
| 	int flags, blocking; | ||||
| 	enum mad_flow result; | ||||
| 
 | ||||
| 	flags = fcntl(fd, F_GETFL); | ||||
| 	if (flags == -1) | ||||
| 	return MAD_FLOW_BREAK; | ||||
| 
 | ||||
| 	blocking = flags & ~O_NONBLOCK; | ||||
| 
 | ||||
| 	if (blocking != flags && | ||||
| 			fcntl(fd, F_SETFL, blocking) == -1) | ||||
| 	return MAD_FLOW_BREAK; | ||||
| 
 | ||||
| 	result = receive_io(fd, buffer, len); | ||||
| 
 | ||||
| 	if (flags != blocking && | ||||
| 			fcntl(fd, F_SETFL, flags) == -1) | ||||
| 	return MAD_FLOW_BREAK; | ||||
| 
 | ||||
| 	return result; | ||||
| } | ||||
| 
 | ||||
| static | ||||
| enum mad_flow ICACHE_FLASH_ATTR send(int fd, void const *message, unsigned int size) | ||||
| { | ||||
| 	enum mad_flow result; | ||||
| 
 | ||||
| 	/* send size */ | ||||
| 
 | ||||
| 	result = send_io(fd, &size, sizeof(size)); | ||||
| 
 | ||||
| 	/* send message */ | ||||
| 
 | ||||
| 	if (result == MAD_FLOW_CONTINUE) | ||||
| 	result = send_io(fd, message, size); | ||||
| 
 | ||||
| 	return result; | ||||
| } | ||||
| 
 | ||||
| static | ||||
| enum mad_flow ICACHE_FLASH_ATTR receive(int fd, void **message, unsigned int *size) | ||||
| { | ||||
| 	enum mad_flow result; | ||||
| 	unsigned int actual; | ||||
| 
 | ||||
| 	if (*message == 0) | ||||
| 	*size = 0; | ||||
| 
 | ||||
| 	/* receive size */ | ||||
| 
 | ||||
| 	result = receive_io(fd, &actual, sizeof(actual)); | ||||
| 
 | ||||
| 	/* receive message */ | ||||
| 
 | ||||
| 	if (result == MAD_FLOW_CONTINUE) { | ||||
| 		if (actual > *size) | ||||
| 		actual -= *size; | ||||
| 		else { | ||||
| 			*size = actual; | ||||
| 			actual = 0; | ||||
| 		} | ||||
| 
 | ||||
| 		if (*size > 0) { | ||||
| 			if (*message == 0) { | ||||
| 				*message = malloc(*size); | ||||
| 				if (*message == 0) | ||||
| 				return MAD_FLOW_BREAK; | ||||
| 			} | ||||
| 
 | ||||
| 			result = receive_io_blocking(fd, *message, *size); | ||||
| 		} | ||||
| 
 | ||||
| 		/* throw away remainder of message */ | ||||
| 
 | ||||
| 		while (actual && result == MAD_FLOW_CONTINUE) { | ||||
| 			char sink[256]; | ||||
| 			unsigned int len; | ||||
| 
 | ||||
| 			len = actual > sizeof(sink) ? sizeof(sink) : actual; | ||||
| 
 | ||||
| 			result = receive_io_blocking(fd, sink, len); | ||||
| 
 | ||||
| 			actual -= len; | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	return result; | ||||
| } | ||||
| 
 | ||||
| static | ||||
| enum mad_flow ICACHE_FLASH_ATTR check_message(struct mad_decoder *decoder) | ||||
| { | ||||
| 	enum mad_flow result; | ||||
| 	void *message = 0; | ||||
| 	unsigned int size; | ||||
| 
 | ||||
| 	result = receive(decoder->async.in, &message, &size); | ||||
| 
 | ||||
| 	if (result == MAD_FLOW_CONTINUE) { | ||||
| 		if (decoder->message_func == 0) | ||||
| 		size = 0; | ||||
| 		else { | ||||
| 			result = decoder->message_func(decoder->cb_data, message, &size); | ||||
| 
 | ||||
| 			if (result == MAD_FLOW_IGNORE || | ||||
| 					result == MAD_FLOW_BREAK) | ||||
| 			size = 0; | ||||
| 		} | ||||
| 
 | ||||
| 		if (send(decoder->async.out, message, size) != MAD_FLOW_CONTINUE) | ||||
| 		result = MAD_FLOW_BREAK; | ||||
| 	} | ||||
| 
 | ||||
| 	if (message) | ||||
| 	free(message); | ||||
| 
 | ||||
| 	return result; | ||||
| } | ||||
| # endif | ||||
| 
 | ||||
| static enum mad_flow ICACHE_FLASH_ATTR error_default(void *data, | ||||
| 		struct mad_stream *stream, struct mad_frame *frame) { | ||||
| 	int *bad_last_frame = data; | ||||
| 
 | ||||
| 	switch (stream->error) { | ||||
| 	case MAD_ERROR_BADCRC: | ||||
| 		if (*bad_last_frame) | ||||
| 			mad_frame_mute(frame); | ||||
| 		else | ||||
| 			*bad_last_frame = 1; | ||||
| 
 | ||||
| 		return MAD_FLOW_IGNORE; | ||||
| 
 | ||||
| 	default: | ||||
| 		return MAD_FLOW_CONTINUE; | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| static | ||||
| int ICACHE_FLASH_ATTR run_sync(struct mad_decoder *decoder) { | ||||
| 	enum mad_flow (*error_func)(void *, struct mad_stream *, struct mad_frame *); | ||||
| 	void *error_data; | ||||
| 	int bad_last_frame = 0; | ||||
| 	struct mad_stream *stream; | ||||
| 	struct mad_frame *frame; | ||||
| 	struct mad_synth *synth; | ||||
| 	int result = 0; | ||||
| 	int r; | ||||
| 
 | ||||
| //	printf("run_sync\n");
 | ||||
| 	if (decoder->input_func == 0) | ||||
| 		return 0; | ||||
| 
 | ||||
| 	if (decoder->error_func) { | ||||
| 		error_func = decoder->error_func; | ||||
| 		error_data = decoder->cb_data; | ||||
| 	} else { | ||||
| 		error_func = error_default; | ||||
| 		error_data = &bad_last_frame; | ||||
| 	} | ||||
| 
 | ||||
| 	stream = &decoder->sync->stream; | ||||
| 	frame = &decoder->sync->frame; | ||||
| 	synth = &decoder->sync->synth; | ||||
| 
 | ||||
| 	mad_stream_init(stream); | ||||
| 	mad_frame_init(frame); | ||||
| 	mad_synth_init(synth); | ||||
| 
 | ||||
| 	mad_stream_options(stream, decoder->options); | ||||
| 
 | ||||
| 	do { | ||||
| 		r = decoder->input_func(decoder->cb_data, stream); | ||||
| //   printf("Input fn: %d\n", r);
 | ||||
| 		switch (r) { | ||||
| 		case MAD_FLOW_STOP: | ||||
| 			goto done; | ||||
| 		case MAD_FLOW_BREAK: | ||||
| 			goto fail; | ||||
| 		case MAD_FLOW_IGNORE: | ||||
| 			continue; | ||||
| 		case MAD_FLOW_CONTINUE: | ||||
| 			break; | ||||
| 		} | ||||
| 
 | ||||
| 		while (1) { | ||||
| # if defined(USE_ASYNC) | ||||
| 			if (decoder->mode == MAD_DECODER_MODE_ASYNC) { | ||||
| 				switch (check_message(decoder)) { | ||||
| 					case MAD_FLOW_IGNORE: | ||||
| 					case MAD_FLOW_CONTINUE: | ||||
| 					break; | ||||
| 					case MAD_FLOW_BREAK: | ||||
| 					goto fail; | ||||
| 					case MAD_FLOW_STOP: | ||||
| 					goto done; | ||||
| 				} | ||||
| 			} | ||||
| # endif | ||||
| 
 | ||||
| 			if (decoder->header_func) { | ||||
| 				r = mad_header_decode(&frame->header, stream); | ||||
| //	printf("mad_header_decode_func: %d\n", r);
 | ||||
| 				if (r != -1) { | ||||
| 					if (!MAD_RECOVERABLE(stream->error)) | ||||
| 						break; | ||||
| 
 | ||||
| 					switch (error_func(error_data, stream, frame)) { | ||||
| 					case MAD_FLOW_STOP: | ||||
| 						goto done; | ||||
| 					case MAD_FLOW_BREAK: | ||||
| 						goto fail; | ||||
| 					case MAD_FLOW_IGNORE: | ||||
| 					case MAD_FLOW_CONTINUE: | ||||
| 					default: | ||||
| 						continue; | ||||
| 					} | ||||
| 				} | ||||
| 
 | ||||
| 				switch (decoder->header_func(decoder->cb_data, &frame->header)) { | ||||
| 				case MAD_FLOW_STOP: | ||||
| 					goto done; | ||||
| 				case MAD_FLOW_BREAK: | ||||
| 					goto fail; | ||||
| 				case MAD_FLOW_IGNORE: | ||||
| 					continue; | ||||
| 				case MAD_FLOW_CONTINUE: | ||||
| 					break; | ||||
| 				} | ||||
| 			} | ||||
| 
 | ||||
| 			r = mad_frame_decode(frame, stream); | ||||
| //	printf("mad_frame_decode: %d\n", r);
 | ||||
| 			if (r == -1) { | ||||
| 				if (!MAD_RECOVERABLE(stream->error)) | ||||
| 					break; | ||||
| 
 | ||||
| 				switch (error_func(error_data, stream, frame)) { | ||||
| 				case MAD_FLOW_STOP: | ||||
| 					goto done; | ||||
| 				case MAD_FLOW_BREAK: | ||||
| 					goto fail; | ||||
| 				case MAD_FLOW_IGNORE: | ||||
| 					break; | ||||
| 				case MAD_FLOW_CONTINUE: | ||||
| 				default: | ||||
| 					continue; | ||||
| 				} | ||||
| 			} else | ||||
| 				bad_last_frame = 0; | ||||
| 
 | ||||
| 			if (decoder->filter_func) { | ||||
| 				switch (decoder->filter_func(decoder->cb_data, stream, frame)) { | ||||
| 				case MAD_FLOW_STOP: | ||||
| 					goto done; | ||||
| 				case MAD_FLOW_BREAK: | ||||
| 					goto fail; | ||||
| 				case MAD_FLOW_IGNORE: | ||||
| 					continue; | ||||
| 				case MAD_FLOW_CONTINUE: | ||||
| 					break; | ||||
| 				} | ||||
| 			} | ||||
| 
 | ||||
| 			mad_synth_frame(synth, frame); | ||||
| 
 | ||||
| //	printf("Calling output fn\n");
 | ||||
| 			if (decoder->output_func) { | ||||
| 				switch (decoder->output_func(decoder->cb_data, &frame->header, | ||||
| 						&synth->pcm)) { | ||||
| 				case MAD_FLOW_STOP: | ||||
| 					goto done; | ||||
| 				case MAD_FLOW_BREAK: | ||||
| 					goto fail; | ||||
| 				case MAD_FLOW_IGNORE: | ||||
| 				case MAD_FLOW_CONTINUE: | ||||
| 					break; | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	} while (stream->error == MAD_ERROR_BUFLEN); | ||||
| 
 | ||||
| 	fail: result = -1; | ||||
| 
 | ||||
| 	done: mad_synth_finish(synth); | ||||
| 	mad_frame_finish(frame); | ||||
| 	mad_stream_finish(stream); | ||||
| 
 | ||||
| 	return result; | ||||
| } | ||||
| 
 | ||||
| # if defined(USE_ASYNC) | ||||
| static | ||||
| int ICACHE_FLASH_ATTR run_async(struct mad_decoder *decoder) | ||||
| { | ||||
| 	pid_t pid; | ||||
| 	int ptoc[2], ctop[2], flags; | ||||
| 
 | ||||
| 	if (pipe(ptoc) == -1) | ||||
| 	return -1; | ||||
| 
 | ||||
| 	if (pipe(ctop) == -1) { | ||||
| 		close(ptoc[0]); | ||||
| 		close(ptoc[1]); | ||||
| 		return -1; | ||||
| 	} | ||||
| 
 | ||||
| 	flags = fcntl(ptoc[0], F_GETFL); | ||||
| 	if (flags == -1 || | ||||
| 			fcntl(ptoc[0], F_SETFL, flags | O_NONBLOCK) == -1) { | ||||
| 		close(ctop[0]); | ||||
| 		close(ctop[1]); | ||||
| 		close(ptoc[0]); | ||||
| 		close(ptoc[1]); | ||||
| 		return -1; | ||||
| 	} | ||||
| 
 | ||||
| 	pid = fork(); | ||||
| 	if (pid == -1) { | ||||
| 		close(ctop[0]); | ||||
| 		close(ctop[1]); | ||||
| 		close(ptoc[0]); | ||||
| 		close(ptoc[1]); | ||||
| 		return -1; | ||||
| 	} | ||||
| 
 | ||||
| 	decoder->async.pid = pid; | ||||
| 
 | ||||
| 	if (pid) { | ||||
| 		/* parent */ | ||||
| 
 | ||||
| 		close(ptoc[0]); | ||||
| 		close(ctop[1]); | ||||
| 
 | ||||
| 		decoder->async.in = ctop[0]; | ||||
| 		decoder->async.out = ptoc[1]; | ||||
| 
 | ||||
| 		return 0; | ||||
| 	} | ||||
| 
 | ||||
| 	/* child */ | ||||
| 
 | ||||
| 	close(ptoc[1]); | ||||
| 	close(ctop[0]); | ||||
| 
 | ||||
| 	decoder->async.in = ptoc[0]; | ||||
| 	decoder->async.out = ctop[1]; | ||||
| 
 | ||||
| 	_exit(run_sync(decoder)); | ||||
| 
 | ||||
| 	/* not reached */ | ||||
| 	return -1; | ||||
| } | ||||
| # endif | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	decoder->run() | ||||
|  * DESCRIPTION:	run the decoder thread either synchronously or asynchronously | ||||
|  */ | ||||
| int ICACHE_FLASH_ATTR mad_decoder_run(struct mad_decoder *decoder, | ||||
| 		enum mad_decoder_mode mode) { | ||||
| 	int result; | ||||
| 	int (*run)(struct mad_decoder *) = 0; | ||||
| //	static struct sync_t decsync; //statically-allocated decoder obj
 | ||||
| 
 | ||||
| 	switch (decoder->mode = mode) { | ||||
| 	case MAD_DECODER_MODE_SYNC: | ||||
| 		run = run_sync; | ||||
| 		break; | ||||
| 
 | ||||
| 	case MAD_DECODER_MODE_ASYNC: | ||||
| # if defined(USE_ASYNC) | ||||
| 		run = run_async; | ||||
| # endif | ||||
| 		break; | ||||
| 	} | ||||
| 
 | ||||
| 	if (run == 0) | ||||
| 		return -1; | ||||
| 
 | ||||
|   decoder->sync = pvPortMalloc(sizeof(*decoder->sync)); | ||||
| //	decoder->sync = &decsync;
 | ||||
| 	if (decoder->sync == 0) | ||||
| 		return -1; | ||||
| 	rtl_memset(decoder->sync, 0, sizeof(*decoder->sync)); | ||||
| 
 | ||||
| 	result = run(decoder); | ||||
| 
 | ||||
|   vPortFree(decoder->sync); | ||||
|   decoder->sync = 0; | ||||
| 
 | ||||
| 	return result; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	decoder->message() | ||||
|  * DESCRIPTION:	send a message to and receive a reply from the decoder process | ||||
|  */ | ||||
| int ICACHE_FLASH_ATTR mad_decoder_message(struct mad_decoder *decoder, | ||||
| 		void *message, unsigned int *len) { | ||||
| # if defined(USE_ASYNC) | ||||
| 	if (decoder->mode != MAD_DECODER_MODE_ASYNC || | ||||
| 			send(decoder->async.out, message, *len) != MAD_FLOW_CONTINUE || | ||||
| 			receive(decoder->async.in, &message, len) != MAD_FLOW_CONTINUE) | ||||
| 	return -1; | ||||
| 
 | ||||
| 	return 0; | ||||
| # else | ||||
| 	return -1; | ||||
| # endif | ||||
| } | ||||
							
								
								
									
										81
									
								
								project/src/mad/fixed.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										81
									
								
								project/src/mad/fixed.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,81 @@ | |||
| /*
 | ||||
|  * libmad - MPEG audio decoder library | ||||
|  * Copyright (C) 2000-2004 Underbit Technologies, Inc. | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|  * the Free Software Foundation; either version 2 of the License, or | ||||
|  * (at your option) any later version. | ||||
|  * | ||||
|  * This program 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.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  * | ||||
|  * You should have received a copy of the GNU General Public License | ||||
|  * along with this program; if not, write to the Free Software | ||||
|  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA | ||||
|  * | ||||
|  * $Id: fixed.c,v 1.13 2004/01/23 09:41:32 rob Exp $ | ||||
|  */ | ||||
| 
 | ||||
| # ifdef HAVE_CONFIG_H | ||||
| #  include "config.h" | ||||
| # endif | ||||
| 
 | ||||
| # include "global.h" | ||||
| 
 | ||||
| # include "fixed.h" | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	fixed->abs() | ||||
|  * DESCRIPTION:	return absolute value of a fixed-point number | ||||
|  */ | ||||
| mad_fixed_t ICACHE_FLASH_ATTR mad_f_abs(mad_fixed_t x) | ||||
| { | ||||
|   return x < 0 ? -x : x; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	fixed->div() | ||||
|  * DESCRIPTION:	perform division using fixed-point math | ||||
|  */ | ||||
| mad_fixed_t ICACHE_FLASH_ATTR mad_f_div(mad_fixed_t x, mad_fixed_t y) | ||||
| { | ||||
|   mad_fixed_t q, r; | ||||
|   unsigned int bits; | ||||
| 
 | ||||
|   q = mad_f_abs(x / y); | ||||
| 
 | ||||
|   if (x < 0) { | ||||
|     x = -x; | ||||
|     y = -y; | ||||
|   } | ||||
| 
 | ||||
|   r = x % y; | ||||
| 
 | ||||
|   if (y < 0) { | ||||
|     x = -x; | ||||
|     y = -y; | ||||
|   } | ||||
| 
 | ||||
|   if (q > mad_f_intpart(MAD_F_MAX) && | ||||
|       !(q == -mad_f_intpart(MAD_F_MIN) && r == 0 && (x < 0) != (y < 0))) | ||||
|     return 0; | ||||
| 
 | ||||
|   for (bits = MAD_F_FRACBITS; bits && r; --bits) { | ||||
|     q <<= 1, r <<= 1; | ||||
|     if (r >= y) | ||||
|       r -= y, ++q; | ||||
|   } | ||||
| 
 | ||||
|   /* round */ | ||||
|   if (2 * r >= y) | ||||
|     ++q; | ||||
| 
 | ||||
|   /* fix sign */ | ||||
|   if ((x < 0) != (y < 0)) | ||||
|     q = -q; | ||||
| 
 | ||||
|   return q << bits; | ||||
| } | ||||
							
								
								
									
										504
									
								
								project/src/mad/frame.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										504
									
								
								project/src/mad/frame.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,504 @@ | |||
| /*
 | ||||
|  * libmad - MPEG audio decoder library | ||||
|  * Copyright (C) 2000-2004 Underbit Technologies, Inc. | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|  * the Free Software Foundation; either version 2 of the License, or | ||||
|  * (at your option) any later version. | ||||
|  * | ||||
|  * This program 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.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  * | ||||
|  * You should have received a copy of the GNU General Public License | ||||
|  * along with this program; if not, write to the Free Software | ||||
|  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA | ||||
|  * | ||||
|  * $Id: frame.c,v 1.29 2004/02/04 22:59:19 rob Exp $ | ||||
|  */ | ||||
| 
 | ||||
| # ifdef HAVE_CONFIG_H | ||||
| #  include "config.h" | ||||
| # endif | ||||
| 
 | ||||
| # include "global.h" | ||||
| 
 | ||||
| # include <stdlib.h> | ||||
| 
 | ||||
| # include "bit.h" | ||||
| # include "stream.h" | ||||
| # include "frame.h" | ||||
| # include "timer.h" | ||||
| //# include "layer12.h"
 | ||||
| # include "layer3.h" | ||||
| 
 | ||||
| static | ||||
| unsigned long const ICACHE_RODATA_ATTR bitrate_table[5][15] = { | ||||
|   /* MPEG-1 */ | ||||
|   { 0,  32000,  64000,  96000, 128000, 160000, 192000, 224000,  /* Layer I   */ | ||||
|        256000, 288000, 320000, 352000, 384000, 416000, 448000 }, | ||||
|   { 0,  32000,  48000,  56000,  64000,  80000,  96000, 112000,  /* Layer II  */ | ||||
|        128000, 160000, 192000, 224000, 256000, 320000, 384000 }, | ||||
|   { 0,  32000,  40000,  48000,  56000,  64000,  80000,  96000,  /* Layer III */ | ||||
|        112000, 128000, 160000, 192000, 224000, 256000, 320000 }, | ||||
| 
 | ||||
|   /* MPEG-2 LSF */ | ||||
|   { 0,  32000,  48000,  56000,  64000,  80000,  96000, 112000,  /* Layer I   */ | ||||
|        128000, 144000, 160000, 176000, 192000, 224000, 256000 }, | ||||
|   { 0,   8000,  16000,  24000,  32000,  40000,  48000,  56000,  /* Layers    */ | ||||
|         64000,  80000,  96000, 112000, 128000, 144000, 160000 } /* II & III  */ | ||||
| }; | ||||
| 
 | ||||
| static | ||||
| unsigned int const ICACHE_RODATA_ATTR samplerate_table[3] = { 44100, 48000, 32000 }; | ||||
| 
 | ||||
| static | ||||
| int ICACHE_RODATA_ATTR (*const decoder_table[3])(struct mad_stream *, struct mad_frame *) = { | ||||
| //  mad_layer_I,
 | ||||
| //  mad_layer_II,
 | ||||
| 	NULL, NULL, | ||||
|   mad_layer_III | ||||
| }; | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	header->init() | ||||
|  * DESCRIPTION:	initialize header struct | ||||
|  */ | ||||
| void ICACHE_FLASH_ATTR mad_header_init(struct mad_header *header) | ||||
| { | ||||
|   header->layer          = 0; | ||||
|   header->mode           = 0; | ||||
|   header->mode_extension = 0; | ||||
|   header->emphasis       = 0; | ||||
| 
 | ||||
|   header->bitrate        = 0; | ||||
|   header->samplerate     = 0; | ||||
| 
 | ||||
|   header->crc_check      = 0; | ||||
|   header->crc_target     = 0; | ||||
| 
 | ||||
|   header->flags          = 0; | ||||
|   header->private_bits   = 0; | ||||
| 
 | ||||
|   header->duration       = mad_timer_zero; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	frame->init() | ||||
|  * DESCRIPTION:	initialize frame struct | ||||
|  */ | ||||
| void ICACHE_FLASH_ATTR mad_frame_init(struct mad_frame *frame) | ||||
| { | ||||
|   mad_header_init(&frame->header); | ||||
| 
 | ||||
|   frame->options = 0; | ||||
| 
 | ||||
|   frame->overlap = 0; | ||||
|   mad_frame_mute(frame); | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	frame->finish() | ||||
|  * DESCRIPTION:	deallocate any dynamic memory associated with frame | ||||
|  */ | ||||
| void ICACHE_FLASH_ATTR mad_frame_finish(struct mad_frame *frame) | ||||
| { | ||||
|   mad_header_finish(&frame->header); | ||||
| 
 | ||||
|   if (frame->overlap) { | ||||
|     vPortFree(frame->overlap); | ||||
|     frame->overlap = 0; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	decode_header() | ||||
|  * DESCRIPTION:	read header data and following CRC word | ||||
|  */ | ||||
| static | ||||
| int ICACHE_FLASH_ATTR decode_header(struct mad_header *header, struct mad_stream *stream) | ||||
| { | ||||
|   unsigned int index; | ||||
| 
 | ||||
|   header->flags        = 0; | ||||
|   header->private_bits = 0; | ||||
| 
 | ||||
|   /* header() */ | ||||
| 
 | ||||
|   /* syncword */ | ||||
|   mad_bit_skip(&stream->ptr, 11); | ||||
| 
 | ||||
|   /* MPEG 2.5 indicator (really part of syncword) */ | ||||
|   if (mad_bit_read(&stream->ptr, 1) == 0) | ||||
|     header->flags |= MAD_FLAG_MPEG_2_5_EXT; | ||||
| 
 | ||||
|   /* ID */ | ||||
|   if (mad_bit_read(&stream->ptr, 1) == 0) | ||||
|     header->flags |= MAD_FLAG_LSF_EXT; | ||||
|   else if (header->flags & MAD_FLAG_MPEG_2_5_EXT) { | ||||
|     stream->error = MAD_ERROR_LOSTSYNC; | ||||
|     return -1; | ||||
|   } | ||||
| 
 | ||||
|   /* layer */ | ||||
|   header->layer = 4 - mad_bit_read(&stream->ptr, 2); | ||||
| 
 | ||||
|   if (header->layer == 4) { | ||||
|     stream->error = MAD_ERROR_BADLAYER; | ||||
|     return -1; | ||||
|   } | ||||
| 
 | ||||
|   /* protection_bit */ | ||||
|   if (mad_bit_read(&stream->ptr, 1) == 0) { | ||||
|     header->flags    |= MAD_FLAG_PROTECTION; | ||||
|     header->crc_check = mad_bit_crc(stream->ptr, 16, 0xffff); | ||||
|   } | ||||
| 
 | ||||
|   /* bitrate_index */ | ||||
|   index = mad_bit_read(&stream->ptr, 4); | ||||
| 
 | ||||
|   if (index == 15) { | ||||
|     stream->error = MAD_ERROR_BADBITRATE; | ||||
|     return -1; | ||||
|   } | ||||
| 
 | ||||
|   if (header->flags & MAD_FLAG_LSF_EXT) | ||||
|     header->bitrate = bitrate_table[3 + (header->layer >> 1)][index]; | ||||
|   else | ||||
|     header->bitrate = bitrate_table[header->layer - 1][index]; | ||||
| 
 | ||||
|   /* sampling_frequency */ | ||||
|   index = mad_bit_read(&stream->ptr, 2); | ||||
| 
 | ||||
|   if (index == 3) { | ||||
|     stream->error = MAD_ERROR_BADSAMPLERATE; | ||||
|     return -1; | ||||
|   } | ||||
| 
 | ||||
|   header->samplerate = samplerate_table[index]; | ||||
| 
 | ||||
|   if (header->flags & MAD_FLAG_LSF_EXT) { | ||||
|     header->samplerate /= 2; | ||||
| 
 | ||||
|     if (header->flags & MAD_FLAG_MPEG_2_5_EXT) | ||||
|       header->samplerate /= 2; | ||||
|   } | ||||
| 
 | ||||
|   /* padding_bit */ | ||||
|   if (mad_bit_read(&stream->ptr, 1)) | ||||
|     header->flags |= MAD_FLAG_PADDING; | ||||
| 
 | ||||
|   /* private_bit */ | ||||
|   if (mad_bit_read(&stream->ptr, 1)) | ||||
|     header->private_bits |= MAD_PRIVATE_HEADER; | ||||
| 
 | ||||
|   /* mode */ | ||||
|   header->mode = 3 - mad_bit_read(&stream->ptr, 2); | ||||
| 
 | ||||
|   /* mode_extension */ | ||||
|   header->mode_extension = mad_bit_read(&stream->ptr, 2); | ||||
| 
 | ||||
|   /* copyright */ | ||||
|   if (mad_bit_read(&stream->ptr, 1)) | ||||
|     header->flags |= MAD_FLAG_COPYRIGHT; | ||||
| 
 | ||||
|   /* original/copy */ | ||||
|   if (mad_bit_read(&stream->ptr, 1)) | ||||
|     header->flags |= MAD_FLAG_ORIGINAL; | ||||
| 
 | ||||
|   /* emphasis */ | ||||
|   header->emphasis = mad_bit_read(&stream->ptr, 2); | ||||
| 
 | ||||
| # if defined(OPT_STRICT) | ||||
|   /*
 | ||||
|    * ISO/IEC 11172-3 says this is a reserved emphasis value, but | ||||
|    * streams exist which use it anyway. Since the value is not important | ||||
|    * to the decoder proper, we allow it unless OPT_STRICT is defined. | ||||
|    */ | ||||
|   if (header->emphasis == MAD_EMPHASIS_RESERVED) { | ||||
|     stream->error = MAD_ERROR_BADEMPHASIS; | ||||
|     return -1; | ||||
|   } | ||||
| # endif | ||||
| 
 | ||||
|   /* error_check() */ | ||||
| 
 | ||||
|   /* crc_check */ | ||||
|   if (header->flags & MAD_FLAG_PROTECTION) | ||||
|     header->crc_target = mad_bit_read(&stream->ptr, 16); | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	free_bitrate() | ||||
|  * DESCRIPTION:	attempt to discover the bitstream's free bitrate | ||||
|  */ | ||||
| static | ||||
| int ICACHE_FLASH_ATTR free_bitrate(struct mad_stream *stream, struct mad_header const *header) | ||||
| { | ||||
|   struct mad_bitptr keep_ptr; | ||||
|   unsigned long rate = 0; | ||||
|   unsigned int pad_slot, slots_per_frame; | ||||
|   unsigned char const *ptr = 0; | ||||
| 
 | ||||
|   keep_ptr = stream->ptr; | ||||
| 
 | ||||
|   pad_slot = (header->flags & MAD_FLAG_PADDING) ? 1 : 0; | ||||
|   slots_per_frame = (header->layer == MAD_LAYER_III && | ||||
| 		     (header->flags & MAD_FLAG_LSF_EXT)) ? 72 : 144; | ||||
| 
 | ||||
|   while (mad_stream_sync(stream) == 0) { | ||||
|     struct mad_stream peek_stream; | ||||
|     struct mad_header peek_header; | ||||
| 
 | ||||
|     peek_stream = *stream; | ||||
|     peek_header = *header; | ||||
| 
 | ||||
|     if (decode_header(&peek_header, &peek_stream) == 0 && | ||||
| 	peek_header.layer == header->layer && | ||||
| 	peek_header.samplerate == header->samplerate) { | ||||
|       unsigned int N; | ||||
| 
 | ||||
|       ptr = mad_bit_nextbyte(&stream->ptr); | ||||
| 
 | ||||
|       N = ptr - stream->this_frame; | ||||
| 
 | ||||
|       if (header->layer == MAD_LAYER_I) { | ||||
| 	rate = (unsigned long) header->samplerate * | ||||
| 	  (N - 4 * pad_slot + 4) / 48 / 1000; | ||||
|       } | ||||
|       else { | ||||
| 	rate = (unsigned long) header->samplerate * | ||||
| 	  (N - pad_slot + 1) / slots_per_frame / 1000; | ||||
|       } | ||||
| 
 | ||||
|       if (rate >= 8) | ||||
| 	break; | ||||
|     } | ||||
| 
 | ||||
|     mad_bit_skip(&stream->ptr, 8); | ||||
|   } | ||||
| 
 | ||||
|   stream->ptr = keep_ptr; | ||||
| 
 | ||||
|   if (rate < 8 || (header->layer == MAD_LAYER_III && rate > 640)) { | ||||
|     stream->error = MAD_ERROR_LOSTSYNC; | ||||
|     return -1; | ||||
|   } | ||||
| 
 | ||||
|   stream->freerate = rate * 1000; | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	header->decode() | ||||
|  * DESCRIPTION:	read the next frame header from the stream | ||||
|  */ | ||||
| int ICACHE_FLASH_ATTR mad_header_decode(struct mad_header *header, struct mad_stream *stream) | ||||
| { | ||||
|   register unsigned char const *ptr, *end; | ||||
|   unsigned int pad_slot, N; | ||||
| 
 | ||||
|   ptr = stream->next_frame; | ||||
|   end = stream->bufend; | ||||
| 
 | ||||
|   if (ptr == 0) { | ||||
|     stream->error = MAD_ERROR_BUFPTR; | ||||
|     goto fail; | ||||
|   } | ||||
| 
 | ||||
|   /* stream skip */ | ||||
|   if (stream->skiplen) { | ||||
|     if (!stream->sync) | ||||
|       ptr = stream->this_frame; | ||||
| 
 | ||||
|     if (end - ptr < stream->skiplen) { | ||||
|       stream->skiplen   -= end - ptr; | ||||
|       stream->next_frame = end; | ||||
| 
 | ||||
|       stream->error = MAD_ERROR_BUFLEN; | ||||
|       goto fail; | ||||
|     } | ||||
| 
 | ||||
|     ptr += stream->skiplen; | ||||
|     stream->skiplen = 0; | ||||
| 
 | ||||
|     stream->sync = 1; | ||||
|   } | ||||
| 
 | ||||
|  sync: | ||||
|   /* synchronize */ | ||||
|   if (stream->sync) { | ||||
|     if (end - ptr < MAD_BUFFER_GUARD) { | ||||
|       stream->next_frame = ptr; | ||||
| 
 | ||||
|       stream->error = MAD_ERROR_BUFLEN; | ||||
|       goto fail; | ||||
|     } | ||||
|     else if (!(ptr[0] == 0xff && (ptr[1] & 0xe0) == 0xe0)) { | ||||
|       /* mark point where frame sync word was expected */ | ||||
|       stream->this_frame = ptr; | ||||
|       stream->next_frame = ptr + 1; | ||||
| 
 | ||||
|       stream->error = MAD_ERROR_LOSTSYNC; | ||||
|       goto fail; | ||||
|     } | ||||
|   } | ||||
|   else { | ||||
|     mad_bit_init(&stream->ptr, ptr); | ||||
| 
 | ||||
|     if (mad_stream_sync(stream) == -1) { | ||||
|       if (end - stream->next_frame >= MAD_BUFFER_GUARD) | ||||
| 	stream->next_frame = end - MAD_BUFFER_GUARD; | ||||
| 
 | ||||
|       stream->error = MAD_ERROR_BUFLEN; | ||||
|       goto fail; | ||||
|     } | ||||
| 
 | ||||
|     ptr = mad_bit_nextbyte(&stream->ptr); | ||||
|   } | ||||
| 
 | ||||
|   /* begin processing */ | ||||
|   stream->this_frame = ptr; | ||||
|   stream->next_frame = ptr + 1;  /* possibly bogus sync word */ | ||||
| 
 | ||||
|   mad_bit_init(&stream->ptr, stream->this_frame); | ||||
| 
 | ||||
|   if (decode_header(header, stream) == -1) | ||||
|     goto fail; | ||||
| 
 | ||||
|   /* calculate frame duration */ | ||||
|   mad_timer_set(&header->duration, 0, | ||||
| 		32 * MAD_NSBSAMPLES(header), header->samplerate); | ||||
| 
 | ||||
|   /* calculate free bit rate */ | ||||
|   if (header->bitrate == 0) { | ||||
|     if ((stream->freerate == 0 || !stream->sync || | ||||
| 	 (header->layer == MAD_LAYER_III && stream->freerate > 640000)) && | ||||
| 	free_bitrate(stream, header) == -1) | ||||
|       goto fail; | ||||
| 
 | ||||
|     header->bitrate = stream->freerate; | ||||
|     header->flags  |= MAD_FLAG_FREEFORMAT; | ||||
|   } | ||||
| 
 | ||||
|   /* calculate beginning of next frame */ | ||||
|   pad_slot = (header->flags & MAD_FLAG_PADDING) ? 1 : 0; | ||||
| 
 | ||||
|   if (header->layer == MAD_LAYER_I) | ||||
|     N = ((12 * header->bitrate / header->samplerate) + pad_slot) * 4; | ||||
|   else { | ||||
|     unsigned int slots_per_frame; | ||||
| 
 | ||||
|     slots_per_frame = (header->layer == MAD_LAYER_III && | ||||
| 		       (header->flags & MAD_FLAG_LSF_EXT)) ? 72 : 144; | ||||
| 
 | ||||
|     N = (slots_per_frame * header->bitrate / header->samplerate) + pad_slot; | ||||
|   } | ||||
| 
 | ||||
|   /* verify there is enough data left in buffer to decode this frame */ | ||||
|   if (N + MAD_BUFFER_GUARD > end - stream->this_frame) { | ||||
|     stream->next_frame = stream->this_frame; | ||||
| 
 | ||||
|     stream->error = MAD_ERROR_BUFLEN; | ||||
|     goto fail; | ||||
|   } | ||||
| 
 | ||||
|   stream->next_frame = stream->this_frame + N; | ||||
| 
 | ||||
|   if (!stream->sync) { | ||||
|     /* check that a valid frame header follows this frame */ | ||||
| 
 | ||||
|     ptr = stream->next_frame; | ||||
|     if (!(ptr[0] == 0xff && (ptr[1] & 0xe0) == 0xe0)) { | ||||
|       ptr = stream->next_frame = stream->this_frame + 1; | ||||
|       goto sync; | ||||
|     } | ||||
| 
 | ||||
|     stream->sync = 1; | ||||
|   } | ||||
| 
 | ||||
|   header->flags |= MAD_FLAG_INCOMPLETE; | ||||
| 
 | ||||
|   return 0; | ||||
| 
 | ||||
|  fail: | ||||
|   stream->sync = 0; | ||||
| 
 | ||||
|   return -1; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	frame->decode() | ||||
|  * DESCRIPTION:	decode a single frame from a bitstream | ||||
|  */ | ||||
| int ICACHE_FLASH_ATTR mad_frame_decode(struct mad_frame *frame, struct mad_stream *stream) | ||||
| { | ||||
|   frame->options = stream->options; | ||||
| 
 | ||||
|   /* header() */ | ||||
|   /* error_check() */ | ||||
| 
 | ||||
|   if (!(frame->header.flags & MAD_FLAG_INCOMPLETE) && | ||||
|       mad_header_decode(&frame->header, stream) == -1) | ||||
|     goto fail; | ||||
| 
 | ||||
|   /* audio_data() */ | ||||
| 
 | ||||
|   frame->header.flags &= ~MAD_FLAG_INCOMPLETE; | ||||
| 
 | ||||
|   if (decoder_table[frame->header.layer - 1](stream, frame) == -1) { | ||||
|     if (!MAD_RECOVERABLE(stream->error)) | ||||
|       stream->next_frame = stream->this_frame; | ||||
| 
 | ||||
|     goto fail; | ||||
|   } | ||||
| 
 | ||||
|   /* ancillary_data() */ | ||||
| 
 | ||||
|   if (frame->header.layer != MAD_LAYER_III) { | ||||
|     struct mad_bitptr next_frame; | ||||
| 
 | ||||
|     mad_bit_init(&next_frame, stream->next_frame); | ||||
| 
 | ||||
|     stream->anc_ptr    = stream->ptr; | ||||
|     stream->anc_bitlen = mad_bit_length(&stream->ptr, &next_frame); | ||||
| 
 | ||||
|     mad_bit_finish(&next_frame); | ||||
|   } | ||||
| 
 | ||||
|   return 0; | ||||
| 
 | ||||
|  fail: | ||||
|   stream->anc_bitlen = 0; | ||||
|   return -1; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	frame->mute() | ||||
|  * DESCRIPTION:	zero all subband values so the frame becomes silent | ||||
|  */ | ||||
| void ICACHE_FLASH_ATTR mad_frame_mute(struct mad_frame *frame) | ||||
| { | ||||
|   unsigned int s, sb; | ||||
| 
 | ||||
|   for (s = 0; s < 36; ++s) { | ||||
|     for (sb = 0; sb < 32; ++sb) { | ||||
|       frame->sbsample[0][s][sb] = | ||||
|       frame->sbsample[1][s][sb] = 0; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   if (frame->overlap) { | ||||
|     for (s = 0; s < 18; ++s) { | ||||
|       for (sb = 0; sb < 32; ++sb) { | ||||
| 	(*frame->overlap)[0][sb][s] = | ||||
| 	(*frame->overlap)[1][sb][s] = 0; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
							
								
								
									
										3109
									
								
								project/src/mad/huffman.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										3109
									
								
								project/src/mad/huffman.c
									
										
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load diff
											
										
									
								
							
							
								
								
									
										2707
									
								
								project/src/mad/layer3.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										2707
									
								
								project/src/mad/layer3.c
									
										
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load diff
											
										
									
								
							
							
								
								
									
										91
									
								
								project/src/mad/mad_version.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										91
									
								
								project/src/mad/mad_version.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,91 @@ | |||
| /*
 | ||||
|  * libmad - MPEG audio decoder library | ||||
|  * Copyright (C) 2000-2004 Underbit Technologies, Inc. | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|  * the Free Software Foundation; either version 2 of the License, or | ||||
|  * (at your option) any later version. | ||||
|  * | ||||
|  * This program 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.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  * | ||||
|  * You should have received a copy of the GNU General Public License | ||||
|  * along with this program; if not, write to the Free Software | ||||
|  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA | ||||
|  * | ||||
|  * $Id: version.c,v 1.15 2004/01/23 09:41:33 rob Exp $ | ||||
|  */ | ||||
| 
 | ||||
| # ifdef HAVE_CONFIG_H | ||||
| #  include "config.h" | ||||
| # endif | ||||
| 
 | ||||
| # include "global.h" | ||||
| 
 | ||||
| # include "mad_version.h" | ||||
| 
 | ||||
| char const mad_version[]   = "MPEG Audio Decoder " MAD_VERSION; | ||||
| char const mad_copyright[] = "Copyright (C) " MAD_PUBLISHYEAR " " MAD_AUTHOR; | ||||
| char const mad_author[]    = MAD_AUTHOR " <" MAD_EMAIL ">"; | ||||
| 
 | ||||
| char const mad_build[] = "" | ||||
| # if defined(DEBUG) | ||||
|   "DEBUG " | ||||
| # elif defined(NDEBUG) | ||||
|   "NDEBUG " | ||||
| # endif | ||||
| 
 | ||||
| # if defined(EXPERIMENTAL) | ||||
|   "EXPERIMENTAL " | ||||
| # endif | ||||
| 
 | ||||
| # if defined(FPM_64BIT) | ||||
|   "FPM_64BIT " | ||||
| # elif defined(FPM_INTEL) | ||||
|   "FPM_INTEL " | ||||
| # elif defined(FPM_ARM) | ||||
|   "FPM_ARM " | ||||
| # elif defined(FPM_MIPS) | ||||
|   "FPM_MIPS " | ||||
| # elif defined(FPM_SPARC) | ||||
|   "FPM_SPARC " | ||||
| # elif defined(FPM_PPC) | ||||
|   "FPM_PPC " | ||||
| # elif defined(FPM_DEFAULT) | ||||
|   "FPM_DEFAULT " | ||||
| # endif | ||||
| 
 | ||||
| # if defined(ASO_IMDCT) | ||||
|   "ASO_IMDCT " | ||||
| # endif | ||||
| # if defined(ASO_INTERLEAVE1) | ||||
|   "ASO_INTERLEAVE1 " | ||||
| # endif | ||||
| # if defined(ASO_INTERLEAVE2) | ||||
|   "ASO_INTERLEAVE2 " | ||||
| # endif | ||||
| # if defined(ASO_ZEROCHECK) | ||||
|   "ASO_ZEROCHECK " | ||||
| # endif | ||||
| 
 | ||||
| # if defined(OPT_SPEED) | ||||
|   "OPT_SPEED " | ||||
| # elif defined(OPT_ACCURACY) | ||||
|   "OPT_ACCURACY " | ||||
| # endif | ||||
| 
 | ||||
| # if defined(OPT_SSO) | ||||
|   "OPT_SSO " | ||||
| # endif | ||||
| 
 | ||||
| # if defined(OPT_DCTO)  /* never defined here */ | ||||
|   "OPT_DCTO " | ||||
| # endif | ||||
| 
 | ||||
| # if defined(OPT_STRICT) | ||||
|   "OPT_STRICT " | ||||
| # endif | ||||
| ; | ||||
							
								
								
									
										534
									
								
								project/src/mad/mpg12/layer12.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										534
									
								
								project/src/mad/mpg12/layer12.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,534 @@ | |||
| /*
 | ||||
|  * libmad - MPEG audio decoder library | ||||
|  * Copyright (C) 2000-2004 Underbit Technologies, Inc. | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|  * the Free Software Foundation; either version 2 of the License, or | ||||
|  * (at your option) any later version. | ||||
|  * | ||||
|  * This program 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.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  * | ||||
|  * You should have received a copy of the GNU General Public License | ||||
|  * along with this program; if not, write to the Free Software | ||||
|  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA | ||||
|  * | ||||
|  * $Id: layer12.c,v 1.17 2004/02/05 09:02:39 rob Exp $ | ||||
|  */ | ||||
| 
 | ||||
| # ifdef HAVE_CONFIG_H | ||||
| #  include "config.h" | ||||
| # endif | ||||
| 
 | ||||
| # include "global.h" | ||||
| 
 | ||||
| # ifdef HAVE_LIMITS_H | ||||
| #  include <limits.h> | ||||
| # else | ||||
| #  define CHAR_BIT  8 | ||||
| # endif | ||||
| 
 | ||||
| # include "fixed.h" | ||||
| # include "bit.h" | ||||
| # include "stream.h" | ||||
| # include "frame.h" | ||||
| # include "mad/mpg12/layer12.h" | ||||
| 
 | ||||
| /*
 | ||||
|  * scalefactor table | ||||
|  * used in both Layer I and Layer II decoding | ||||
|  */ | ||||
| static | ||||
| mad_fixed_t const ICACHE_RODATA_ATTR sf_table[64] = { | ||||
| # include "sf_table.dat" | ||||
| }; | ||||
| 
 | ||||
| /* --- Layer I ------------------------------------------------------------- */ | ||||
| 
 | ||||
| /* linear scaling table */ | ||||
| static | ||||
| mad_fixed_t const ICACHE_RODATA_ATTR linear_table[14] = { | ||||
|   MAD_F(0x15555555),  /* 2^2  / (2^2  - 1) == 1.33333333333333 */ | ||||
|   MAD_F(0x12492492),  /* 2^3  / (2^3  - 1) == 1.14285714285714 */ | ||||
|   MAD_F(0x11111111),  /* 2^4  / (2^4  - 1) == 1.06666666666667 */ | ||||
|   MAD_F(0x10842108),  /* 2^5  / (2^5  - 1) == 1.03225806451613 */ | ||||
|   MAD_F(0x10410410),  /* 2^6  / (2^6  - 1) == 1.01587301587302 */ | ||||
|   MAD_F(0x10204081),  /* 2^7  / (2^7  - 1) == 1.00787401574803 */ | ||||
|   MAD_F(0x10101010),  /* 2^8  / (2^8  - 1) == 1.00392156862745 */ | ||||
|   MAD_F(0x10080402),  /* 2^9  / (2^9  - 1) == 1.00195694716243 */ | ||||
|   MAD_F(0x10040100),  /* 2^10 / (2^10 - 1) == 1.00097751710655 */ | ||||
|   MAD_F(0x10020040),  /* 2^11 / (2^11 - 1) == 1.00048851978505 */ | ||||
|   MAD_F(0x10010010),  /* 2^12 / (2^12 - 1) == 1.00024420024420 */ | ||||
|   MAD_F(0x10008004),  /* 2^13 / (2^13 - 1) == 1.00012208521548 */ | ||||
|   MAD_F(0x10004001),  /* 2^14 / (2^14 - 1) == 1.00006103888177 */ | ||||
|   MAD_F(0x10002000)   /* 2^15 / (2^15 - 1) == 1.00003051850948 */ | ||||
| }; | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	I_sample() | ||||
|  * DESCRIPTION:	decode one requantized Layer I sample from a bitstream | ||||
|  */ | ||||
| static | ||||
| mad_fixed_t ICACHE_FLASH_ATTR I_sample(struct mad_bitptr *ptr, unsigned int nb) | ||||
| { | ||||
|   mad_fixed_t sample; | ||||
| 
 | ||||
|   sample = mad_bit_read(ptr, nb); | ||||
| 
 | ||||
|   /* invert most significant bit, extend sign, then scale to fixed format */ | ||||
| 
 | ||||
|   sample ^= 1 << (nb - 1); | ||||
|   sample |= -(sample & (1 << (nb - 1))); | ||||
| 
 | ||||
|   sample <<= MAD_F_FRACBITS - (nb - 1); | ||||
| 
 | ||||
|   /* requantize the sample */ | ||||
| 
 | ||||
|   /* s'' = (2^nb / (2^nb - 1)) * (s''' + 2^(-nb + 1)) */ | ||||
| 
 | ||||
|   sample += MAD_F_ONE >> (nb - 1); | ||||
| 
 | ||||
|   return mad_f_mul(sample, linear_table[nb - 2]); | ||||
| 
 | ||||
|   /* s' = factor * s'' */ | ||||
|   /* (to be performed by caller) */ | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	layer->I() | ||||
|  * DESCRIPTION:	decode a single Layer I frame | ||||
|  */ | ||||
| int ICACHE_FLASH_ATTR mad_layer_I(struct mad_stream *stream, struct mad_frame *frame) | ||||
| { | ||||
|   struct mad_header *header = &frame->header; | ||||
|   unsigned int nch, bound, ch, s, sb, nb; | ||||
|   unsigned char allocation[2][32], scalefactor[2][32]; | ||||
| 
 | ||||
|   nch = MAD_NCHANNELS(header); | ||||
| 
 | ||||
|   bound = 32; | ||||
|   if (header->mode == MAD_MODE_JOINT_STEREO) { | ||||
|     header->flags |= MAD_FLAG_I_STEREO; | ||||
|     bound = 4 + header->mode_extension * 4; | ||||
|   } | ||||
| 
 | ||||
|   /* check CRC word */ | ||||
| 
 | ||||
|   if (header->flags & MAD_FLAG_PROTECTION) { | ||||
|     header->crc_check = | ||||
|       mad_bit_crc(stream->ptr, 4 * (bound * nch + (32 - bound)), | ||||
| 		  header->crc_check); | ||||
| 
 | ||||
|     if (header->crc_check != header->crc_target && | ||||
| 	!(frame->options & MAD_OPTION_IGNORECRC)) { | ||||
|       stream->error = MAD_ERROR_BADCRC; | ||||
|       return -1; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   /* decode bit allocations */ | ||||
| 
 | ||||
|   for (sb = 0; sb < bound; ++sb) { | ||||
|     for (ch = 0; ch < nch; ++ch) { | ||||
|       nb = mad_bit_read(&stream->ptr, 4); | ||||
| 
 | ||||
|       if (nb == 15) { | ||||
| 	stream->error = MAD_ERROR_BADBITALLOC; | ||||
| 	return -1; | ||||
|       } | ||||
| 
 | ||||
|       allocation[ch][sb] = nb ? nb + 1 : 0; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   for (sb = bound; sb < 32; ++sb) { | ||||
|     nb = mad_bit_read(&stream->ptr, 4); | ||||
| 
 | ||||
|     if (nb == 15) { | ||||
|       stream->error = MAD_ERROR_BADBITALLOC; | ||||
|       return -1; | ||||
|     } | ||||
| 
 | ||||
|     allocation[0][sb] = | ||||
|     allocation[1][sb] = nb ? nb + 1 : 0; | ||||
|   } | ||||
| 
 | ||||
|   /* decode scalefactors */ | ||||
| 
 | ||||
|   for (sb = 0; sb < 32; ++sb) { | ||||
|     for (ch = 0; ch < nch; ++ch) { | ||||
|       if (allocation[ch][sb]) { | ||||
| 	scalefactor[ch][sb] = mad_bit_read(&stream->ptr, 6); | ||||
| 
 | ||||
| # if defined(OPT_STRICT) | ||||
| 	/*
 | ||||
| 	 * Scalefactor index 63 does not appear in Table B.1 of | ||||
| 	 * ISO/IEC 11172-3. Nonetheless, other implementations accept it, | ||||
| 	 * so we only reject it if OPT_STRICT is defined. | ||||
| 	 */ | ||||
| 	if (scalefactor[ch][sb] == 63) { | ||||
| 	  stream->error = MAD_ERROR_BADSCALEFACTOR; | ||||
| 	  return -1; | ||||
| 	} | ||||
| # endif | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   /* decode samples */ | ||||
| 
 | ||||
|   for (s = 0; s < 12; ++s) { | ||||
|     for (sb = 0; sb < bound; ++sb) { | ||||
|       for (ch = 0; ch < nch; ++ch) { | ||||
| 	nb = allocation[ch][sb]; | ||||
| 	frame->sbsample[ch][s][sb] = nb ? | ||||
| 	  mad_f_mul(I_sample(&stream->ptr, nb), | ||||
| 		    sf_table[scalefactor[ch][sb]]) : 0; | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     for (sb = bound; sb < 32; ++sb) { | ||||
|       if ((nb = allocation[0][sb])) { | ||||
| 	mad_fixed_t sample; | ||||
| 
 | ||||
| 	sample = I_sample(&stream->ptr, nb); | ||||
| 
 | ||||
| 	for (ch = 0; ch < nch; ++ch) { | ||||
| 	  frame->sbsample[ch][s][sb] = | ||||
| 	    mad_f_mul(sample, sf_table[scalefactor[ch][sb]]); | ||||
| 	} | ||||
|       } | ||||
|       else { | ||||
| 	for (ch = 0; ch < nch; ++ch) | ||||
| 	  frame->sbsample[ch][s][sb] = 0; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| /* --- Layer II ------------------------------------------------------------ */ | ||||
| 
 | ||||
| /* possible quantization per subband table */ | ||||
| static | ||||
| struct { | ||||
|   unsigned int sblimit; | ||||
|   unsigned char const offsets[30]; | ||||
| } const ICACHE_RODATA_ATTR sbquant_table[5] = { | ||||
|   /* ISO/IEC 11172-3 Table B.2a */ | ||||
|   { 27, { 7, 7, 7, 6, 6, 6, 6, 6, 6, 6, 6, 3, 3, 3, 3, 3,	/* 0 */ | ||||
| 	  3, 3, 3, 3, 3, 3, 3, 0, 0, 0, 0 } }, | ||||
|   /* ISO/IEC 11172-3 Table B.2b */ | ||||
|   { 30, { 7, 7, 7, 6, 6, 6, 6, 6, 6, 6, 6, 3, 3, 3, 3, 3,	/* 1 */ | ||||
| 	  3, 3, 3, 3, 3, 3, 3, 0, 0, 0, 0, 0, 0, 0 } }, | ||||
|   /* ISO/IEC 11172-3 Table B.2c */ | ||||
|   {  8, { 5, 5, 2, 2, 2, 2, 2, 2 } },				/* 2 */ | ||||
|   /* ISO/IEC 11172-3 Table B.2d */ | ||||
|   { 12, { 5, 5, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2 } },		/* 3 */ | ||||
|   /* ISO/IEC 13818-3 Table B.1 */ | ||||
|   { 30, { 4, 4, 4, 4, 2, 2, 2, 2, 2, 2, 2, 1, 1, 1, 1, 1,	/* 4 */ | ||||
| 	  1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1 } } | ||||
| }; | ||||
| 
 | ||||
| /* bit allocation table */ | ||||
| static | ||||
| struct { | ||||
|   unsigned short nbal; | ||||
|   unsigned short offset; | ||||
| } const ICACHE_RODATA_ATTR bitalloc_table[8] = { | ||||
|   { 2, 0 },  /* 0 */ | ||||
|   { 2, 3 },  /* 1 */ | ||||
|   { 3, 3 },  /* 2 */ | ||||
|   { 3, 1 },  /* 3 */ | ||||
|   { 4, 2 },  /* 4 */ | ||||
|   { 4, 3 },  /* 5 */ | ||||
|   { 4, 4 },  /* 6 */ | ||||
|   { 4, 5 }   /* 7 */ | ||||
| }; | ||||
| 
 | ||||
| /* offsets into quantization class table */ | ||||
| static | ||||
| unsigned char const ICACHE_RODATA_ATTR offset_table[6][15] = { | ||||
|   { 0, 1, 16                                             },  /* 0 */ | ||||
|   { 0, 1,  2, 3, 4, 5, 16                                },  /* 1 */ | ||||
|   { 0, 1,  2, 3, 4, 5,  6, 7,  8,  9, 10, 11, 12, 13, 14 },  /* 2 */ | ||||
|   { 0, 1,  3, 4, 5, 6,  7, 8,  9, 10, 11, 12, 13, 14, 15 },  /* 3 */ | ||||
|   { 0, 1,  2, 3, 4, 5,  6, 7,  8,  9, 10, 11, 12, 13, 16 },  /* 4 */ | ||||
|   { 0, 2,  4, 5, 6, 7,  8, 9, 10, 11, 12, 13, 14, 15, 16 }   /* 5 */ | ||||
| }; | ||||
| 
 | ||||
| /* quantization class table */ | ||||
| static | ||||
| struct quantclass { | ||||
|   unsigned short nlevels; | ||||
|   unsigned char group; | ||||
|   unsigned char bits; | ||||
|   mad_fixed_t C; | ||||
|   mad_fixed_t D; | ||||
| } const qc_table[17] = { | ||||
| # include "qc_table.dat" | ||||
| }; | ||||
| //ICACHE_RODATA_ATTR 
 | ||||
| /*
 | ||||
|  * NAME:	II_samples() | ||||
|  * DESCRIPTION:	decode three requantized Layer II samples from a bitstream | ||||
|  */ | ||||
| static | ||||
| void ICACHE_FLASH_ATTR II_samples(struct mad_bitptr *ptr, | ||||
| 		struct quantclass const *quantclass, | ||||
| 		mad_fixed_t output[3]) | ||||
| { | ||||
|   unsigned int nb, s, sample[3]; | ||||
| 
 | ||||
|   if ((nb = quantclass->group)) { | ||||
|     unsigned int c, nlevels; | ||||
| 
 | ||||
|     /* degrouping */ | ||||
|     c = mad_bit_read(ptr, quantclass->bits); | ||||
|     nlevels = quantclass->nlevels; | ||||
| 
 | ||||
|     for (s = 0; s < 3; ++s) { | ||||
|       sample[s] = c % nlevels; | ||||
|       c /= nlevels; | ||||
|     } | ||||
|   } | ||||
|   else { | ||||
|     nb = quantclass->bits; | ||||
| 
 | ||||
|     for (s = 0; s < 3; ++s) | ||||
|       sample[s] = mad_bit_read(ptr, nb); | ||||
|   } | ||||
| 
 | ||||
|   for (s = 0; s < 3; ++s) { | ||||
|     mad_fixed_t requantized; | ||||
| 
 | ||||
|     /* invert most significant bit, extend sign, then scale to fixed format */ | ||||
| 
 | ||||
|     requantized  = sample[s] ^ (1 << (nb - 1)); | ||||
|     requantized |= -(requantized & (1 << (nb - 1))); | ||||
| 
 | ||||
|     requantized <<= MAD_F_FRACBITS - (nb - 1); | ||||
| 
 | ||||
|     /* requantize the sample */ | ||||
| 
 | ||||
|     /* s'' = C * (s''' + D) */ | ||||
| 
 | ||||
|     output[s] = mad_f_mul(requantized + quantclass->D, quantclass->C); | ||||
| 
 | ||||
|     /* s' = factor * s'' */ | ||||
|     /* (to be performed by caller) */ | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	layer->II() | ||||
|  * DESCRIPTION:	decode a single Layer II frame | ||||
|  */ | ||||
| int ICACHE_FLASH_ATTR mad_layer_II(struct mad_stream *stream, struct mad_frame *frame) | ||||
| { | ||||
|   struct mad_header *header = &frame->header; | ||||
|   struct mad_bitptr start; | ||||
|   unsigned int index, sblimit, nbal, nch, bound, gr, ch, s, sb; | ||||
|   unsigned char const *offsets; | ||||
|   unsigned char allocation[2][32], scfsi[2][32], scalefactor[2][32][3]; | ||||
|   mad_fixed_t samples[3]; | ||||
| 
 | ||||
|   nch = MAD_NCHANNELS(header); | ||||
| 
 | ||||
|   if (header->flags & MAD_FLAG_LSF_EXT) | ||||
|     index = 4; | ||||
|   else if (header->flags & MAD_FLAG_FREEFORMAT) | ||||
|     goto freeformat; | ||||
|   else { | ||||
|     unsigned long bitrate_per_channel; | ||||
| 
 | ||||
|     bitrate_per_channel = header->bitrate; | ||||
|     if (nch == 2) { | ||||
|       bitrate_per_channel /= 2; | ||||
| 
 | ||||
| # if defined(OPT_STRICT) | ||||
|       /*
 | ||||
|        * ISO/IEC 11172-3 allows only single channel mode for 32, 48, 56, and | ||||
|        * 80 kbps bitrates in Layer II, but some encoders ignore this | ||||
|        * restriction. We enforce it if OPT_STRICT is defined. | ||||
|        */ | ||||
|       if (bitrate_per_channel <= 28000 || bitrate_per_channel == 40000) { | ||||
| 	stream->error = MAD_ERROR_BADMODE; | ||||
| 	return -1; | ||||
|       } | ||||
| # endif | ||||
|     } | ||||
|     else {  /* nch == 1 */ | ||||
|       if (bitrate_per_channel > 192000) { | ||||
| 	/*
 | ||||
| 	 * ISO/IEC 11172-3 does not allow single channel mode for 224, 256, | ||||
| 	 * 320, or 384 kbps bitrates in Layer II. | ||||
| 	 */ | ||||
| 	stream->error = MAD_ERROR_BADMODE; | ||||
| 	return -1; | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     if (bitrate_per_channel <= 48000) | ||||
|       index = (header->samplerate == 32000) ? 3 : 2; | ||||
|     else if (bitrate_per_channel <= 80000) | ||||
|       index = 0; | ||||
|     else { | ||||
|     freeformat: | ||||
|       index = (header->samplerate == 48000) ? 0 : 1; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   sblimit = sbquant_table[index].sblimit; | ||||
|   offsets = sbquant_table[index].offsets; | ||||
| 
 | ||||
|   bound = 32; | ||||
|   if (header->mode == MAD_MODE_JOINT_STEREO) { | ||||
|     header->flags |= MAD_FLAG_I_STEREO; | ||||
|     bound = 4 + header->mode_extension * 4; | ||||
|   } | ||||
| 
 | ||||
|   if (bound > sblimit) | ||||
|     bound = sblimit; | ||||
| 
 | ||||
|   start = stream->ptr; | ||||
| 
 | ||||
|   /* decode bit allocations */ | ||||
| 
 | ||||
|   for (sb = 0; sb < bound; ++sb) { | ||||
|     nbal = bitalloc_table[offsets[sb]].nbal; | ||||
| 
 | ||||
|     for (ch = 0; ch < nch; ++ch) | ||||
|       allocation[ch][sb] = mad_bit_read(&stream->ptr, nbal); | ||||
|   } | ||||
| 
 | ||||
|   for (sb = bound; sb < sblimit; ++sb) { | ||||
|     nbal = bitalloc_table[offsets[sb]].nbal; | ||||
| 
 | ||||
|     allocation[0][sb] = | ||||
|     allocation[1][sb] = mad_bit_read(&stream->ptr, nbal); | ||||
|   } | ||||
| 
 | ||||
|   /* decode scalefactor selection info */ | ||||
| 
 | ||||
|   for (sb = 0; sb < sblimit; ++sb) { | ||||
|     for (ch = 0; ch < nch; ++ch) { | ||||
|       if (allocation[ch][sb]) | ||||
| 	scfsi[ch][sb] = mad_bit_read(&stream->ptr, 2); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   /* check CRC word */ | ||||
| 
 | ||||
|   if (header->flags & MAD_FLAG_PROTECTION) { | ||||
|     header->crc_check = | ||||
|       mad_bit_crc(start, mad_bit_length(&start, &stream->ptr), | ||||
| 		  header->crc_check); | ||||
| 
 | ||||
|     if (header->crc_check != header->crc_target && | ||||
| 	!(frame->options & MAD_OPTION_IGNORECRC)) { | ||||
|       stream->error = MAD_ERROR_BADCRC; | ||||
|       return -1; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   /* decode scalefactors */ | ||||
| 
 | ||||
|   for (sb = 0; sb < sblimit; ++sb) { | ||||
|     for (ch = 0; ch < nch; ++ch) { | ||||
|       if (allocation[ch][sb]) { | ||||
| 	scalefactor[ch][sb][0] = mad_bit_read(&stream->ptr, 6); | ||||
| 
 | ||||
| 	switch (scfsi[ch][sb]) { | ||||
| 	case 2: | ||||
| 	  scalefactor[ch][sb][2] = | ||||
| 	  scalefactor[ch][sb][1] = | ||||
| 	  scalefactor[ch][sb][0]; | ||||
| 	  break; | ||||
| 
 | ||||
| 	case 0: | ||||
| 	  scalefactor[ch][sb][1] = mad_bit_read(&stream->ptr, 6); | ||||
| 	  /* fall through */ | ||||
| 
 | ||||
| 	case 1: | ||||
| 	case 3: | ||||
| 	  scalefactor[ch][sb][2] = mad_bit_read(&stream->ptr, 6); | ||||
| 	} | ||||
| 
 | ||||
| 	if (scfsi[ch][sb] & 1) | ||||
| 	  scalefactor[ch][sb][1] = scalefactor[ch][sb][scfsi[ch][sb] - 1]; | ||||
| 
 | ||||
| # if defined(OPT_STRICT) | ||||
| 	/*
 | ||||
| 	 * Scalefactor index 63 does not appear in Table B.1 of | ||||
| 	 * ISO/IEC 11172-3. Nonetheless, other implementations accept it, | ||||
| 	 * so we only reject it if OPT_STRICT is defined. | ||||
| 	 */ | ||||
| 	if (scalefactor[ch][sb][0] == 63 || | ||||
| 	    scalefactor[ch][sb][1] == 63 || | ||||
| 	    scalefactor[ch][sb][2] == 63) { | ||||
| 	  stream->error = MAD_ERROR_BADSCALEFACTOR; | ||||
| 	  return -1; | ||||
| 	} | ||||
| # endif | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   /* decode samples */ | ||||
| 
 | ||||
|   for (gr = 0; gr < 12; ++gr) { | ||||
|     for (sb = 0; sb < bound; ++sb) { | ||||
|       for (ch = 0; ch < nch; ++ch) { | ||||
| 	if ((index = allocation[ch][sb])) { | ||||
| 	  index = offset_table[bitalloc_table[offsets[sb]].offset][index - 1]; | ||||
| 
 | ||||
| 	  II_samples(&stream->ptr, &qc_table[index], samples); | ||||
| 
 | ||||
| 	  for (s = 0; s < 3; ++s) { | ||||
| 	    frame->sbsample[ch][3 * gr + s][sb] = | ||||
| 	      mad_f_mul(samples[s], sf_table[scalefactor[ch][sb][gr / 4]]); | ||||
| 	  } | ||||
| 	} | ||||
| 	else { | ||||
| 	  for (s = 0; s < 3; ++s) | ||||
| 	    frame->sbsample[ch][3 * gr + s][sb] = 0; | ||||
| 	} | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     for (sb = bound; sb < sblimit; ++sb) { | ||||
|       if ((index = allocation[0][sb])) { | ||||
| 	index = offset_table[bitalloc_table[offsets[sb]].offset][index - 1]; | ||||
| 
 | ||||
| 	II_samples(&stream->ptr, &qc_table[index], samples); | ||||
| 
 | ||||
| 	for (ch = 0; ch < nch; ++ch) { | ||||
| 	  for (s = 0; s < 3; ++s) { | ||||
| 	    frame->sbsample[ch][3 * gr + s][sb] = | ||||
| 	      mad_f_mul(samples[s], sf_table[scalefactor[ch][sb][gr / 4]]); | ||||
| 	  } | ||||
| 	} | ||||
|       } | ||||
|       else { | ||||
| 	for (ch = 0; ch < nch; ++ch) { | ||||
| 	  for (s = 0; s < 3; ++s) | ||||
| 	    frame->sbsample[ch][3 * gr + s][sb] = 0; | ||||
| 	} | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     for (ch = 0; ch < nch; ++ch) { | ||||
|       for (s = 0; s < 3; ++s) { | ||||
| 	for (sb = sblimit; sb < 32; ++sb) | ||||
| 	  frame->sbsample[ch][3 * gr + s][sb] = 0; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
							
								
								
									
										2
									
								
								project/src/mad/mpg12/readme.txt
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								project/src/mad/mpg12/readme.txt
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,2 @@ | |||
| Because of size constraints, the option to read mpg1/2 files has been | ||||
| disabled. | ||||
							
								
								
									
										163
									
								
								project/src/mad/stream.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										163
									
								
								project/src/mad/stream.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,163 @@ | |||
| /*
 | ||||
|  * libmad - MPEG audio decoder library | ||||
|  * Copyright (C) 2000-2004 Underbit Technologies, Inc. | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|  * the Free Software Foundation; either version 2 of the License, or | ||||
|  * (at your option) any later version. | ||||
|  * | ||||
|  * This program 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.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  * | ||||
|  * You should have received a copy of the GNU General Public License | ||||
|  * along with this program; if not, write to the Free Software | ||||
|  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA | ||||
|  * | ||||
|  * $Id: stream.c,v 1.12 2004/02/05 09:02:39 rob Exp $ | ||||
|  */ | ||||
| 
 | ||||
| # ifdef HAVE_CONFIG_H | ||||
| #  include "config.h" | ||||
| # endif | ||||
| 
 | ||||
| # include "global.h" | ||||
| 
 | ||||
| # include <stdlib.h> | ||||
| 
 | ||||
| # include "bit.h" | ||||
| # include "stream.h" | ||||
| 
 | ||||
| //main_data_t MainData; //static alloc of decoder data
 | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	stream->init() | ||||
|  * DESCRIPTION:	initialize stream struct | ||||
|  */ | ||||
| void ICACHE_FLASH_ATTR mad_stream_init(struct mad_stream *stream) | ||||
| { | ||||
|   stream->buffer     = 0; | ||||
|   stream->bufend     = 0; | ||||
|   stream->skiplen    = 0; | ||||
| 
 | ||||
|   stream->sync       = 0; | ||||
|   stream->freerate   = 0; | ||||
| 
 | ||||
|   stream->this_frame = 0; | ||||
|   stream->next_frame = 0; | ||||
|   mad_bit_init(&stream->ptr, 0); | ||||
| 
 | ||||
|   mad_bit_init(&stream->anc_ptr, 0); | ||||
|   stream->anc_bitlen = 0; | ||||
| 
 | ||||
|   stream->main_data  = 0; | ||||
|   stream->md_len     = 0; | ||||
| 
 | ||||
|   stream->options    = 0; | ||||
|   stream->error      = MAD_ERROR_NONE; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	stream->finish() | ||||
|  * DESCRIPTION:	deallocate any dynamic memory associated with stream | ||||
|  */ | ||||
| void ICACHE_FLASH_ATTR mad_stream_finish(struct mad_stream *stream) | ||||
| { | ||||
|   if (stream->main_data) { | ||||
|     vPortFree(stream->main_data); | ||||
|     stream->main_data = 0; | ||||
|   } | ||||
| 
 | ||||
|   mad_bit_finish(&stream->anc_ptr); | ||||
|   mad_bit_finish(&stream->ptr); | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	stream->buffer() | ||||
|  * DESCRIPTION:	set stream buffer pointers | ||||
|  */ | ||||
| void ICACHE_FLASH_ATTR mad_stream_buffer(struct mad_stream *stream, | ||||
| 		       unsigned char const *buffer, unsigned long length) | ||||
| { | ||||
|   stream->buffer = buffer; | ||||
|   stream->bufend = buffer + length; | ||||
| 
 | ||||
|   stream->this_frame = buffer; | ||||
|   stream->next_frame = buffer; | ||||
| 
 | ||||
|   stream->sync = 1; | ||||
| 
 | ||||
|   mad_bit_init(&stream->ptr, buffer); | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	stream->skip() | ||||
|  * DESCRIPTION:	arrange to skip bytes before the next frame | ||||
|  */ | ||||
| void ICACHE_FLASH_ATTR mad_stream_skip(struct mad_stream *stream, unsigned long length) | ||||
| { | ||||
|   stream->skiplen += length; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	stream->sync() | ||||
|  * DESCRIPTION:	locate the next stream sync word | ||||
|  */ | ||||
| int ICACHE_FLASH_ATTR mad_stream_sync(struct mad_stream *stream) | ||||
| { | ||||
|   register unsigned char const *ptr, *end; | ||||
| 
 | ||||
|   ptr = mad_bit_nextbyte(&stream->ptr); | ||||
|   end = stream->bufend; | ||||
| 
 | ||||
|   while (ptr < end - 1 && | ||||
| 	 !(ptr[0] == 0xff && (ptr[1] & 0xe0) == 0xe0)) | ||||
|     ++ptr; | ||||
| 
 | ||||
|   if (end - ptr < MAD_BUFFER_GUARD) | ||||
|     return -1; | ||||
| 
 | ||||
|   mad_bit_init(&stream->ptr, ptr); | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	stream->errorstr() | ||||
|  * DESCRIPTION:	return a string description of the current error condition | ||||
|  */ | ||||
| char const ICACHE_FLASH_ATTR *mad_stream_errorstr(struct mad_stream const *stream) | ||||
| { | ||||
|   switch (stream->error) { | ||||
|   case MAD_ERROR_NONE:		 return "no error"; | ||||
| 
 | ||||
|   case MAD_ERROR_BUFLEN:	 return "input buffer too small (or EOF)"; | ||||
|   case MAD_ERROR_BUFPTR:	 return "invalid (null) buffer pointer"; | ||||
| 
 | ||||
|   case MAD_ERROR_NOMEM:		 return "not enough memory"; | ||||
| 
 | ||||
|   case MAD_ERROR_LOSTSYNC:	 return "lost synchronization"; | ||||
|   case MAD_ERROR_BADLAYER:	 return "reserved header layer value"; | ||||
|   case MAD_ERROR_BADBITRATE:	 return "forbidden bitrate value"; | ||||
|   case MAD_ERROR_BADSAMPLERATE:	 return "reserved sample frequency value"; | ||||
|   case MAD_ERROR_BADEMPHASIS:	 return "reserved emphasis value"; | ||||
| 
 | ||||
|   case MAD_ERROR_BADCRC:	 return "CRC check failed"; | ||||
|   case MAD_ERROR_BADBITALLOC:	 return "forbidden bit allocation value"; | ||||
|   case MAD_ERROR_BADSCALEFACTOR: return "bad scalefactor index"; | ||||
|   case MAD_ERROR_BADMODE:	 return "bad bitrate/mode combination"; | ||||
|   case MAD_ERROR_BADFRAMELEN:	 return "bad frame length"; | ||||
|   case MAD_ERROR_BADBIGVALUES:	 return "bad big_values count"; | ||||
|   case MAD_ERROR_BADBLOCKTYPE:	 return "reserved block_type"; | ||||
|   case MAD_ERROR_BADSCFSI:	 return "bad scalefactor selection info"; | ||||
|   case MAD_ERROR_BADDATAPTR:	 return "bad main_data_begin pointer"; | ||||
|   case MAD_ERROR_BADPART3LEN:	 return "bad audio data length"; | ||||
|   case MAD_ERROR_BADHUFFTABLE:	 return "bad Huffman table select"; | ||||
|   case MAD_ERROR_BADHUFFDATA:	 return "Huffman data overrun"; | ||||
|   case MAD_ERROR_BADSTEREO:	 return "incompatible block_type for JS"; | ||||
|   } | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
							
								
								
									
										934
									
								
								project/src/mad/synth.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										934
									
								
								project/src/mad/synth.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,934 @@ | |||
| /*
 | ||||
|  * libmad - MPEG audio decoder library | ||||
|  * Copyright (C) 2000-2004 Underbit Technologies, Inc. | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|  * the Free Software Foundation; either version 2 of the License, or | ||||
|  * (at your option) any later version. | ||||
|  * | ||||
|  * This program 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.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  * | ||||
|  * You should have received a copy of the GNU General Public License | ||||
|  * along with this program; if not, write to the Free Software | ||||
|  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA | ||||
|  * | ||||
|  * $Id: synth.c,v 1.25 2004/01/23 09:41:33 rob Exp $ | ||||
|  */ | ||||
| 
 | ||||
| # ifdef HAVE_CONFIG_H | ||||
| #  include "config.h" | ||||
| # endif | ||||
| 
 | ||||
| # include "global.h" | ||||
| 
 | ||||
| # include "fixed.h" | ||||
| # include "frame.h" | ||||
| # include "synth.h" | ||||
| # include "string.h" | ||||
| 
 | ||||
| // #define SAVED_SAMPLE_BUFF_LEN   240000
 | ||||
| // unsigned int saved_idx = 0;
 | ||||
| // short int saved_samples[SAVED_SAMPLE_BUFF_LEN];
 | ||||
| 
 | ||||
| /*
 | ||||
|  * The following utility routine performs simple rounding, clipping, and | ||||
|  * scaling of MAD's high-resolution samples down to 16 bits. It does not | ||||
|  * perform any dithering or noise shaping, which would be recommended to | ||||
|  * obtain any exceptional audio quality. It is therefore not recommended to | ||||
|  * use this routine if high-quality output is desired. | ||||
|  */ | ||||
| 
 | ||||
| static inline | ||||
| signed int scale(mad_fixed_t sample) | ||||
| { | ||||
|   /* round */ | ||||
|   sample += (1L << (MAD_F_FRACBITS - 16)); | ||||
| 
 | ||||
|   /* clip */ | ||||
|   if (sample >= MAD_F_ONE) | ||||
|     sample = MAD_F_ONE - 1; | ||||
|   else if (sample < -MAD_F_ONE) | ||||
|     sample = -MAD_F_ONE; | ||||
| 
 | ||||
|   /* quantize */ | ||||
|   return sample >> (MAD_F_FRACBITS + 1 - 16); | ||||
| } | ||||
| /*
 | ||||
|  * NAME:	synth->init() | ||||
|  * DESCRIPTION:	initialize synth struct | ||||
|  */ | ||||
| void mad_synth_init(struct mad_synth *synth) | ||||
| { | ||||
|   mad_synth_mute(synth); | ||||
| 
 | ||||
|   synth->phase = 0; | ||||
| 
 | ||||
|   synth->pcm.samplerate = 0; | ||||
|   synth->pcm.channels   = 0; | ||||
|   synth->pcm.length     = 0;   | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	synth->mute() | ||||
|  * DESCRIPTION:	zero all polyphase filterbank values, resetting synthesis | ||||
|  */ | ||||
| void mad_synth_mute(struct mad_synth *synth) | ||||
| { | ||||
|   unsigned int ch, s, v; | ||||
| 
 | ||||
|   for (ch = 0; ch < 2; ++ch) { | ||||
|     for (s = 0; s < 16; ++s) { | ||||
|       for (v = 0; v < 8; ++v) { | ||||
|             synth->filter[ch][0][0][s][v] = synth->filter[ch][0][1][s][v] = | ||||
| 	    synth->filter[ch][1][0][s][v] = synth->filter[ch][1][1][s][v] = 0; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * An optional optimization called here the Subband Synthesis Optimization | ||||
|  * (SSO) improves the performance of subband synthesis at the expense of | ||||
|  * accuracy. | ||||
|  * | ||||
|  * The idea is to simplify 32x32->64-bit multiplication to 32x32->32 such | ||||
|  * that extra scaling and rounding are not necessary. This often allows the | ||||
|  * compiler to use faster 32-bit multiply-accumulate instructions instead of | ||||
|  * explicit 64-bit multiply, shift, and add instructions. | ||||
|  * | ||||
|  * SSO works like this: a full 32x32->64-bit multiply of two mad_fixed_t | ||||
|  * values requires the result to be right-shifted 28 bits to be properly | ||||
|  * scaled to the same fixed-point format. Right shifts can be applied at any | ||||
|  * time to either operand or to the result, so the optimization involves | ||||
|  * careful placement of these shifts to minimize the loss of accuracy. | ||||
|  * | ||||
|  * First, a 14-bit shift is applied with rounding at compile-time to the D[] | ||||
|  * table of coefficients for the subband synthesis window. This only loses 2 | ||||
|  * bits of accuracy because the lower 12 bits are always zero. A second | ||||
|  * 12-bit shift occurs after the DCT calculation. This loses 12 bits of | ||||
|  * accuracy. Finally, a third 2-bit shift occurs just before the sample is | ||||
|  * saved in the PCM buffer. 14 + 12 + 2 == 28 bits. | ||||
|  */ | ||||
| 
 | ||||
| /* FPM_DEFAULT without OPT_SSO will actually lose accuracy and performance */ | ||||
| 
 | ||||
| # if defined(FPM_DEFAULT) && !defined(OPT_SSO) | ||||
| #  define OPT_SSO | ||||
| # endif | ||||
| 
 | ||||
| /* second SSO shift, with rounding */ | ||||
| 
 | ||||
| # if defined(OPT_SSO) | ||||
| #  define SHIFT(x)  (((x) + (1L << 11)) >> 12) | ||||
| # else | ||||
| #  define SHIFT(x)  (x) | ||||
| # endif | ||||
| 
 | ||||
| /* possible DCT speed optimization */ | ||||
| 
 | ||||
| # if defined(OPT_SPEED) && defined(MAD_F_MLX) | ||||
| #  define OPT_DCTO | ||||
| #  define MUL(x, y)  \ | ||||
|     ({ mad_fixed64hi_t hi;  \ | ||||
|        mad_fixed64lo_t lo;  \ | ||||
|        MAD_F_MLX(hi, lo, (x), (y));  \ | ||||
|        hi << (32 - MAD_F_SCALEBITS - 3);  \ | ||||
|     }) | ||||
| # else | ||||
| #  undef OPT_DCTO | ||||
| #  define MUL(x, y)  mad_f_mul((x), (y)) | ||||
| # endif | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	dct32() | ||||
|  * DESCRIPTION:	perform fast in[32]->out[32] DCT | ||||
|  */ | ||||
| static | ||||
| void dct32(mad_fixed_t const in[32], unsigned int slot, | ||||
| 	   mad_fixed_t lo[16][8], mad_fixed_t hi[16][8]) | ||||
| { | ||||
|   mad_fixed_t t0,   t1,   t2,   t3,   t4,   t5,   t6,   t7; | ||||
|   mad_fixed_t t8,   t9,   t10,  t11,  t12,  t13,  t14,  t15; | ||||
|   mad_fixed_t t16,  t17,  t18,  t19,  t20,  t21,  t22,  t23; | ||||
|   mad_fixed_t t24,  t25,  t26,  t27,  t28,  t29,  t30,  t31; | ||||
|   mad_fixed_t t32,  t33,  t34,  t35,  t36,  t37,  t38,  t39; | ||||
|   mad_fixed_t t40,  t41,  t42,  t43,  t44,  t45,  t46,  t47; | ||||
|   mad_fixed_t t48,  t49,  t50,  t51,  t52,  t53,  t54,  t55; | ||||
|   mad_fixed_t t56,  t57,  t58,  t59,  t60,  t61,  t62,  t63; | ||||
|   mad_fixed_t t64,  t65,  t66,  t67,  t68,  t69,  t70,  t71; | ||||
|   mad_fixed_t t72,  t73,  t74,  t75,  t76,  t77,  t78,  t79; | ||||
|   mad_fixed_t t80,  t81,  t82,  t83,  t84,  t85,  t86,  t87; | ||||
|   mad_fixed_t t88,  t89,  t90,  t91,  t92,  t93,  t94,  t95; | ||||
|   mad_fixed_t t96,  t97,  t98,  t99,  t100, t101, t102, t103; | ||||
|   mad_fixed_t t104, t105, t106, t107, t108, t109, t110, t111; | ||||
|   mad_fixed_t t112, t113, t114, t115, t116, t117, t118, t119; | ||||
|   mad_fixed_t t120, t121, t122, t123, t124, t125, t126, t127; | ||||
|   mad_fixed_t t128, t129, t130, t131, t132, t133, t134, t135; | ||||
|   mad_fixed_t t136, t137, t138, t139, t140, t141, t142, t143; | ||||
|   mad_fixed_t t144, t145, t146, t147, t148, t149, t150, t151; | ||||
|   mad_fixed_t t152, t153, t154, t155, t156, t157, t158, t159; | ||||
|   mad_fixed_t t160, t161, t162, t163, t164, t165, t166, t167; | ||||
|   mad_fixed_t t168, t169, t170, t171, t172, t173, t174, t175; | ||||
|   mad_fixed_t t176; | ||||
| 
 | ||||
|   /* costab[i] = cos(PI / (2 * 32) * i) */ | ||||
| 
 | ||||
| # if defined(OPT_DCTO) | ||||
| #  define costab1	MAD_F(0x7fd8878e) | ||||
| #  define costab2	MAD_F(0x7f62368f) | ||||
| #  define costab3	MAD_F(0x7e9d55fc) | ||||
| #  define costab4	MAD_F(0x7d8a5f40) | ||||
| #  define costab5	MAD_F(0x7c29fbee) | ||||
| #  define costab6	MAD_F(0x7a7d055b) | ||||
| #  define costab7	MAD_F(0x78848414) | ||||
| #  define costab8	MAD_F(0x7641af3d) | ||||
| #  define costab9	MAD_F(0x73b5ebd1) | ||||
| #  define costab10	MAD_F(0x70e2cbc6) | ||||
| #  define costab11	MAD_F(0x6dca0d14) | ||||
| #  define costab12	MAD_F(0x6a6d98a4) | ||||
| #  define costab13	MAD_F(0x66cf8120) | ||||
| #  define costab14	MAD_F(0x62f201ac) | ||||
| #  define costab15	MAD_F(0x5ed77c8a) | ||||
| #  define costab16	MAD_F(0x5a82799a) | ||||
| #  define costab17	MAD_F(0x55f5a4d2) | ||||
| #  define costab18	MAD_F(0x5133cc94) | ||||
| #  define costab19	MAD_F(0x4c3fdff4) | ||||
| #  define costab20	MAD_F(0x471cece7) | ||||
| #  define costab21	MAD_F(0x41ce1e65) | ||||
| #  define costab22	MAD_F(0x3c56ba70) | ||||
| #  define costab23	MAD_F(0x36ba2014) | ||||
| #  define costab24	MAD_F(0x30fbc54d) | ||||
| #  define costab25	MAD_F(0x2b1f34eb) | ||||
| #  define costab26	MAD_F(0x25280c5e) | ||||
| #  define costab27	MAD_F(0x1f19f97b) | ||||
| #  define costab28	MAD_F(0x18f8b83c) | ||||
| #  define costab29	MAD_F(0x12c8106f) | ||||
| #  define costab30	MAD_F(0x0c8bd35e) | ||||
| #  define costab31	MAD_F(0x0647d97c) | ||||
| # else | ||||
| #  define costab1	MAD_F(0x0ffb10f2)  /* 0.998795456 */ | ||||
| #  define costab2	MAD_F(0x0fec46d2)  /* 0.995184727 */ | ||||
| #  define costab3	MAD_F(0x0fd3aac0)  /* 0.989176510 */ | ||||
| #  define costab4	MAD_F(0x0fb14be8)  /* 0.980785280 */ | ||||
| #  define costab5	MAD_F(0x0f853f7e)  /* 0.970031253 */ | ||||
| #  define costab6	MAD_F(0x0f4fa0ab)  /* 0.956940336 */ | ||||
| #  define costab7	MAD_F(0x0f109082)  /* 0.941544065 */ | ||||
| #  define costab8	MAD_F(0x0ec835e8)  /* 0.923879533 */ | ||||
| #  define costab9	MAD_F(0x0e76bd7a)  /* 0.903989293 */ | ||||
| #  define costab10	MAD_F(0x0e1c5979)  /* 0.881921264 */ | ||||
| #  define costab11	MAD_F(0x0db941a3)  /* 0.857728610 */ | ||||
| #  define costab12	MAD_F(0x0d4db315)  /* 0.831469612 */ | ||||
| #  define costab13	MAD_F(0x0cd9f024)  /* 0.803207531 */ | ||||
| #  define costab14	MAD_F(0x0c5e4036)  /* 0.773010453 */ | ||||
| #  define costab15	MAD_F(0x0bdaef91)  /* 0.740951125 */ | ||||
| #  define costab16	MAD_F(0x0b504f33)  /* 0.707106781 */ | ||||
| #  define costab17	MAD_F(0x0abeb49a)  /* 0.671558955 */ | ||||
| #  define costab18	MAD_F(0x0a267993)  /* 0.634393284 */ | ||||
| #  define costab19	MAD_F(0x0987fbfe)  /* 0.595699304 */ | ||||
| #  define costab20	MAD_F(0x08e39d9d)  /* 0.555570233 */ | ||||
| #  define costab21	MAD_F(0x0839c3cd)  /* 0.514102744 */ | ||||
| #  define costab22	MAD_F(0x078ad74e)  /* 0.471396737 */ | ||||
| #  define costab23	MAD_F(0x06d74402)  /* 0.427555093 */ | ||||
| #  define costab24	MAD_F(0x061f78aa)  /* 0.382683432 */ | ||||
| #  define costab25	MAD_F(0x0563e69d)  /* 0.336889853 */ | ||||
| #  define costab26	MAD_F(0x04a5018c)  /* 0.290284677 */ | ||||
| #  define costab27	MAD_F(0x03e33f2f)  /* 0.242980180 */ | ||||
| #  define costab28	MAD_F(0x031f1708)  /* 0.195090322 */ | ||||
| #  define costab29	MAD_F(0x0259020e)  /* 0.146730474 */ | ||||
| #  define costab30	MAD_F(0x01917a6c)  /* 0.098017140 */ | ||||
| #  define costab31	MAD_F(0x00c8fb30)  /* 0.049067674 */ | ||||
| # endif | ||||
| 
 | ||||
|   t0   = in[0]  + in[31];  t16  = MUL(in[0]  - in[31], costab1); | ||||
|   t1   = in[15] + in[16];  t17  = MUL(in[15] - in[16], costab31); | ||||
| 
 | ||||
|   t41  = t16 + t17; | ||||
|   t59  = MUL(t16 - t17, costab2); | ||||
|   t33  = t0  + t1; | ||||
|   t50  = MUL(t0  - t1,  costab2); | ||||
| 
 | ||||
|   t2   = in[7]  + in[24];  t18  = MUL(in[7]  - in[24], costab15); | ||||
|   t3   = in[8]  + in[23];  t19  = MUL(in[8]  - in[23], costab17); | ||||
| 
 | ||||
|   t42  = t18 + t19; | ||||
|   t60  = MUL(t18 - t19, costab30); | ||||
|   t34  = t2  + t3; | ||||
|   t51  = MUL(t2  - t3,  costab30); | ||||
| 
 | ||||
|   t4   = in[3]  + in[28];  t20  = MUL(in[3]  - in[28], costab7); | ||||
|   t5   = in[12] + in[19];  t21  = MUL(in[12] - in[19], costab25); | ||||
| 
 | ||||
|   t43  = t20 + t21; | ||||
|   t61  = MUL(t20 - t21, costab14); | ||||
|   t35  = t4  + t5; | ||||
|   t52  = MUL(t4  - t5,  costab14); | ||||
| 
 | ||||
|   t6   = in[4]  + in[27];  t22  = MUL(in[4]  - in[27], costab9); | ||||
|   t7   = in[11] + in[20];  t23  = MUL(in[11] - in[20], costab23); | ||||
| 
 | ||||
|   t44  = t22 + t23; | ||||
|   t62  = MUL(t22 - t23, costab18); | ||||
|   t36  = t6  + t7; | ||||
|   t53  = MUL(t6  - t7,  costab18); | ||||
| 
 | ||||
|   t8   = in[1]  + in[30];  t24  = MUL(in[1]  - in[30], costab3); | ||||
|   t9   = in[14] + in[17];  t25  = MUL(in[14] - in[17], costab29); | ||||
| 
 | ||||
|   t45  = t24 + t25; | ||||
|   t63  = MUL(t24 - t25, costab6); | ||||
|   t37  = t8  + t9; | ||||
|   t54  = MUL(t8  - t9,  costab6); | ||||
| 
 | ||||
|   t10  = in[6]  + in[25];  t26  = MUL(in[6]  - in[25], costab13); | ||||
|   t11  = in[9]  + in[22];  t27  = MUL(in[9]  - in[22], costab19); | ||||
| 
 | ||||
|   t46  = t26 + t27; | ||||
|   t64  = MUL(t26 - t27, costab26); | ||||
|   t38  = t10 + t11; | ||||
|   t55  = MUL(t10 - t11, costab26); | ||||
| 
 | ||||
|   t12  = in[2]  + in[29];  t28  = MUL(in[2]  - in[29], costab5); | ||||
|   t13  = in[13] + in[18];  t29  = MUL(in[13] - in[18], costab27); | ||||
| 
 | ||||
|   t47  = t28 + t29; | ||||
|   t65  = MUL(t28 - t29, costab10); | ||||
|   t39  = t12 + t13; | ||||
|   t56  = MUL(t12 - t13, costab10); | ||||
| 
 | ||||
|   t14  = in[5]  + in[26];  t30  = MUL(in[5]  - in[26], costab11); | ||||
|   t15  = in[10] + in[21];  t31  = MUL(in[10] - in[21], costab21); | ||||
| 
 | ||||
|   t48  = t30 + t31; | ||||
|   t66  = MUL(t30 - t31, costab22); | ||||
|   t40  = t14 + t15; | ||||
|   t57  = MUL(t14 - t15, costab22); | ||||
| 
 | ||||
|   t69  = t33 + t34;  t89  = MUL(t33 - t34, costab4); | ||||
|   t70  = t35 + t36;  t90  = MUL(t35 - t36, costab28); | ||||
|   t71  = t37 + t38;  t91  = MUL(t37 - t38, costab12); | ||||
|   t72  = t39 + t40;  t92  = MUL(t39 - t40, costab20); | ||||
|   t73  = t41 + t42;  t94  = MUL(t41 - t42, costab4); | ||||
|   t74  = t43 + t44;  t95  = MUL(t43 - t44, costab28); | ||||
|   t75  = t45 + t46;  t96  = MUL(t45 - t46, costab12); | ||||
|   t76  = t47 + t48;  t97  = MUL(t47 - t48, costab20); | ||||
| 
 | ||||
|   t78  = t50 + t51;  t100 = MUL(t50 - t51, costab4); | ||||
|   t79  = t52 + t53;  t101 = MUL(t52 - t53, costab28); | ||||
|   t80  = t54 + t55;  t102 = MUL(t54 - t55, costab12); | ||||
|   t81  = t56 + t57;  t103 = MUL(t56 - t57, costab20); | ||||
| 
 | ||||
|   t83  = t59 + t60;  t106 = MUL(t59 - t60, costab4); | ||||
|   t84  = t61 + t62;  t107 = MUL(t61 - t62, costab28); | ||||
|   t85  = t63 + t64;  t108 = MUL(t63 - t64, costab12); | ||||
|   t86  = t65 + t66;  t109 = MUL(t65 - t66, costab20); | ||||
| 
 | ||||
|   t113 = t69  + t70; | ||||
|   t114 = t71  + t72; | ||||
| 
 | ||||
|   /*  0 */ hi[15][slot] = SHIFT(t113 + t114); | ||||
|   /* 16 */ lo[ 0][slot] = SHIFT(MUL(t113 - t114, costab16)); | ||||
| 
 | ||||
|   t115 = t73  + t74; | ||||
|   t116 = t75  + t76; | ||||
| 
 | ||||
|   t32  = t115 + t116; | ||||
| 
 | ||||
|   /*  1 */ hi[14][slot] = SHIFT(t32); | ||||
| 
 | ||||
|   t118 = t78  + t79; | ||||
|   t119 = t80  + t81; | ||||
| 
 | ||||
|   t58  = t118 + t119; | ||||
| 
 | ||||
|   /*  2 */ hi[13][slot] = SHIFT(t58); | ||||
| 
 | ||||
|   t121 = t83  + t84; | ||||
|   t122 = t85  + t86; | ||||
| 
 | ||||
|   t67  = t121 + t122; | ||||
| 
 | ||||
|   t49  = (t67 * 2) - t32; | ||||
| 
 | ||||
|   /*  3 */ hi[12][slot] = SHIFT(t49); | ||||
| 
 | ||||
|   t125 = t89  + t90; | ||||
|   t126 = t91  + t92; | ||||
| 
 | ||||
|   t93  = t125 + t126; | ||||
| 
 | ||||
|   /*  4 */ hi[11][slot] = SHIFT(t93); | ||||
| 
 | ||||
|   t128 = t94  + t95; | ||||
|   t129 = t96  + t97; | ||||
| 
 | ||||
|   t98  = t128 + t129; | ||||
| 
 | ||||
|   t68  = (t98 * 2) - t49; | ||||
| 
 | ||||
|   /*  5 */ hi[10][slot] = SHIFT(t68); | ||||
| 
 | ||||
|   t132 = t100 + t101; | ||||
|   t133 = t102 + t103; | ||||
| 
 | ||||
|   t104 = t132 + t133; | ||||
| 
 | ||||
|   t82  = (t104 * 2) - t58; | ||||
| 
 | ||||
|   /*  6 */ hi[ 9][slot] = SHIFT(t82); | ||||
| 
 | ||||
|   t136 = t106 + t107; | ||||
|   t137 = t108 + t109; | ||||
| 
 | ||||
|   t110 = t136 + t137; | ||||
| 
 | ||||
|   t87  = (t110 * 2) - t67; | ||||
| 
 | ||||
|   t77  = (t87 * 2) - t68; | ||||
| 
 | ||||
|   /*  7 */ hi[ 8][slot] = SHIFT(t77); | ||||
| 
 | ||||
|   t141 = MUL(t69 - t70, costab8); | ||||
|   t142 = MUL(t71 - t72, costab24); | ||||
|   t143 = t141 + t142; | ||||
| 
 | ||||
|   /*  8 */ hi[ 7][slot] = SHIFT(t143); | ||||
|   /* 24 */ lo[ 8][slot] = | ||||
| 	     SHIFT((MUL(t141 - t142, costab16) * 2) - t143); | ||||
| 
 | ||||
|   t144 = MUL(t73 - t74, costab8); | ||||
|   t145 = MUL(t75 - t76, costab24); | ||||
|   t146 = t144 + t145; | ||||
| 
 | ||||
|   t88  = (t146 * 2) - t77; | ||||
| 
 | ||||
|   /*  9 */ hi[ 6][slot] = SHIFT(t88); | ||||
| 
 | ||||
|   t148 = MUL(t78 - t79, costab8); | ||||
|   t149 = MUL(t80 - t81, costab24); | ||||
|   t150 = t148 + t149; | ||||
| 
 | ||||
|   t105 = (t150 * 2) - t82; | ||||
| 
 | ||||
|   /* 10 */ hi[ 5][slot] = SHIFT(t105); | ||||
| 
 | ||||
|   t152 = MUL(t83 - t84, costab8); | ||||
|   t153 = MUL(t85 - t86, costab24); | ||||
|   t154 = t152 + t153; | ||||
| 
 | ||||
|   t111 = (t154 * 2) - t87; | ||||
| 
 | ||||
|   t99  = (t111 * 2) - t88; | ||||
| 
 | ||||
|   /* 11 */ hi[ 4][slot] = SHIFT(t99); | ||||
| 
 | ||||
|   t157 = MUL(t89 - t90, costab8); | ||||
|   t158 = MUL(t91 - t92, costab24); | ||||
|   t159 = t157 + t158; | ||||
| 
 | ||||
|   t127 = (t159 * 2) - t93; | ||||
| 
 | ||||
|   /* 12 */ hi[ 3][slot] = SHIFT(t127); | ||||
| 
 | ||||
|   t160 = (MUL(t125 - t126, costab16) * 2) - t127; | ||||
| 
 | ||||
|   /* 20 */ lo[ 4][slot] = SHIFT(t160); | ||||
|   /* 28 */ lo[12][slot] = | ||||
| 	     SHIFT((((MUL(t157 - t158, costab16) * 2) - t159) * 2) - t160); | ||||
| 
 | ||||
|   t161 = MUL(t94 - t95, costab8); | ||||
|   t162 = MUL(t96 - t97, costab24); | ||||
|   t163 = t161 + t162; | ||||
| 
 | ||||
|   t130 = (t163 * 2) - t98; | ||||
| 
 | ||||
|   t112 = (t130 * 2) - t99; | ||||
| 
 | ||||
|   /* 13 */ hi[ 2][slot] = SHIFT(t112); | ||||
| 
 | ||||
|   t164 = (MUL(t128 - t129, costab16) * 2) - t130; | ||||
| 
 | ||||
|   t166 = MUL(t100 - t101, costab8); | ||||
|   t167 = MUL(t102 - t103, costab24); | ||||
|   t168 = t166 + t167; | ||||
| 
 | ||||
|   t134 = (t168 * 2) - t104; | ||||
| 
 | ||||
|   t120 = (t134 * 2) - t105; | ||||
| 
 | ||||
|   /* 14 */ hi[ 1][slot] = SHIFT(t120); | ||||
| 
 | ||||
|   t135 = (MUL(t118 - t119, costab16) * 2) - t120; | ||||
| 
 | ||||
|   /* 18 */ lo[ 2][slot] = SHIFT(t135); | ||||
| 
 | ||||
|   t169 = (MUL(t132 - t133, costab16) * 2) - t134; | ||||
| 
 | ||||
|   t151 = (t169 * 2) - t135; | ||||
| 
 | ||||
|   /* 22 */ lo[ 6][slot] = SHIFT(t151); | ||||
| 
 | ||||
|   t170 = (((MUL(t148 - t149, costab16) * 2) - t150) * 2) - t151; | ||||
| 
 | ||||
|   /* 26 */ lo[10][slot] = SHIFT(t170); | ||||
|   /* 30 */ lo[14][slot] = | ||||
| 	     SHIFT((((((MUL(t166 - t167, costab16) * 2) - | ||||
| 		       t168) * 2) - t169) * 2) - t170); | ||||
| 
 | ||||
|   t171 = MUL(t106 - t107, costab8); | ||||
|   t172 = MUL(t108 - t109, costab24); | ||||
|   t173 = t171 + t172; | ||||
| 
 | ||||
|   t138 = (t173 * 2) - t110; | ||||
| 
 | ||||
|   t123 = (t138 * 2) - t111; | ||||
| 
 | ||||
|   t139 = (MUL(t121 - t122, costab16) * 2) - t123; | ||||
| 
 | ||||
|   t117 = (t123 * 2) - t112; | ||||
| 
 | ||||
|   /* 15 */ hi[ 0][slot] = SHIFT(t117); | ||||
| 
 | ||||
|   t124 = (MUL(t115 - t116, costab16) * 2) - t117; | ||||
| 
 | ||||
|   /* 17 */ lo[ 1][slot] = SHIFT(t124); | ||||
| 
 | ||||
|   t131 = (t139 * 2) - t124; | ||||
| 
 | ||||
|   /* 19 */ lo[ 3][slot] = SHIFT(t131); | ||||
| 
 | ||||
|   t140 = (t164 * 2) - t131; | ||||
| 
 | ||||
|   /* 21 */ lo[ 5][slot] = SHIFT(t140); | ||||
| 
 | ||||
|   t174 = (MUL(t136 - t137, costab16) * 2) - t138; | ||||
| 
 | ||||
|   t155 = (t174 * 2) - t139; | ||||
| 
 | ||||
|   t147 = (t155 * 2) - t140; | ||||
| 
 | ||||
|   /* 23 */ lo[ 7][slot] = SHIFT(t147); | ||||
| 
 | ||||
|   t156 = (((MUL(t144 - t145, costab16) * 2) - t146) * 2) - t147; | ||||
| 
 | ||||
|   /* 25 */ lo[ 9][slot] = SHIFT(t156); | ||||
| 
 | ||||
|   t175 = (((MUL(t152 - t153, costab16) * 2) - t154) * 2) - t155; | ||||
| 
 | ||||
|   t165 = (t175 * 2) - t156; | ||||
| 
 | ||||
|   /* 27 */ lo[11][slot] = SHIFT(t165); | ||||
| 
 | ||||
|   t176 = (((((MUL(t161 - t162, costab16) * 2) - | ||||
| 	     t163) * 2) - t164) * 2) - t165; | ||||
| 
 | ||||
|   /* 29 */ lo[13][slot] = SHIFT(t176); | ||||
|   /* 31 */ lo[15][slot] = | ||||
| 	     SHIFT((((((((MUL(t171 - t172, costab16) * 2) - | ||||
| 			 t173) * 2) - t174) * 2) - t175) * 2) - t176); | ||||
| 
 | ||||
|   /*
 | ||||
|    * Totals: | ||||
|    *  80 multiplies | ||||
|    *  80 additions | ||||
|    * 119 subtractions | ||||
|    *  49 shifts (not counting SSO) | ||||
|    */ | ||||
| } | ||||
| 
 | ||||
| # undef MUL | ||||
| # undef SHIFT | ||||
| 
 | ||||
| /* third SSO shift and/or D[] optimization preshift */ | ||||
| 
 | ||||
| # if defined(OPT_SSO) | ||||
| #  if MAD_F_FRACBITS != 28 | ||||
| #   error "MAD_F_FRACBITS must be 28 to use OPT_SSO" | ||||
| #  endif | ||||
| #  define ML0(hi, lo, x, y)	((lo)  = (x) * (y)) | ||||
| #  define MLA(hi, lo, x, y)	((lo) += (x) * (y)) | ||||
| #  define MLN(hi, lo)		((lo)  = -(lo)) | ||||
| #  define MLZ(hi, lo)		((void) (hi), (mad_fixed_t) (lo)) | ||||
| #  define SHIFT(x)		((x) >> 2) | ||||
| #  define PRESHIFT(x)		((MAD_F(x) + (1L << 13)) >> 14) | ||||
| # else | ||||
| #  define ML0(hi, lo, x, y)	MAD_F_ML0((hi), (lo), (x), (y)) | ||||
| #  define MLA(hi, lo, x, y)	MAD_F_MLA((hi), (lo), (x), (y)) | ||||
| #  define MLN(hi, lo)		MAD_F_MLN((hi), (lo)) | ||||
| #  define MLZ(hi, lo)		MAD_F_MLZ((hi), (lo)) | ||||
| #  define SHIFT(x)		(x) | ||||
| #  if defined(MAD_F_SCALEBITS) | ||||
| #   undef  MAD_F_SCALEBITS | ||||
| #   define MAD_F_SCALEBITS	(MAD_F_FRACBITS - 12) | ||||
| #   define PRESHIFT(x)		(MAD_F(x) >> 12) | ||||
| #  else | ||||
| #   define PRESHIFT(x)		MAD_F(x) | ||||
| #  endif | ||||
| # endif | ||||
| 
 | ||||
| static | ||||
| mad_fixed_t const D[17][32] = { | ||||
| # include "D.dat" | ||||
| }; | ||||
| 
 | ||||
| # if defined(ASO_SYNTH) | ||||
| void synth_full(struct mad_synth *, struct mad_frame const *, | ||||
| 		unsigned int, unsigned int); | ||||
| # else | ||||
| /*
 | ||||
|  * NAME:	synth->full() | ||||
|  * DESCRIPTION:	perform full frequency PCM synthesis | ||||
|  */ | ||||
| static | ||||
| void synth_full(struct mad_synth *synth, struct mad_frame const *frame, | ||||
| 		unsigned int nch, unsigned int ns) | ||||
| { | ||||
|   unsigned int phase, ch, s, sb, pe, po; | ||||
|   short int *pcm1, *pcm2; | ||||
|   mad_fixed_t (*filter)[2][2][16][8]; | ||||
|   mad_fixed_t (*sbsample)[36][32]; | ||||
|   register mad_fixed_t (*fe)[8], (*fx)[8], (*fo)[8]; | ||||
|   register mad_fixed_t const (*Dptr)[32], *ptr ; | ||||
|   register mad_fixed64hi_t hi; | ||||
|   register mad_fixed64lo_t lo; | ||||
|   mad_fixed_t raw_sample; | ||||
|   short int short_sample_buff[2][32]; | ||||
| 
 | ||||
|   phase = synth->phase; | ||||
| 
 | ||||
|   if (nch > 2) | ||||
|    return; | ||||
| 
 | ||||
|   for (s = 0; s < ns; ++s) | ||||
|   { | ||||
|     memset (short_sample_buff, 0x00, sizeof(short_sample_buff)); | ||||
| 
 | ||||
|     for (ch = 0; ch < nch; ++ch) | ||||
|     { | ||||
|       sbsample = &frame->sbsample[ch]; | ||||
|       filter   = &synth->filter[ch]; | ||||
|       pcm1     = short_sample_buff[ch]; | ||||
| 
 | ||||
|       dct32((*sbsample)[s], phase >> 1, | ||||
| 	    (*filter)[0][phase & 1], (*filter)[1][phase & 1]); | ||||
| 
 | ||||
|       pe = phase & ~1; | ||||
|       po = ((phase - 1) & 0xf) | 1; | ||||
| 
 | ||||
|       /* calculate 16 samples */ | ||||
| 
 | ||||
|       fe = &(*filter)[0][ phase & 1][0]; | ||||
|       fx = &(*filter)[0][~phase & 1][0]; | ||||
|       fo = &(*filter)[1][~phase & 1][0]; | ||||
| 
 | ||||
|       Dptr = &D[0]; | ||||
| 
 | ||||
|       ptr = *Dptr + po; | ||||
|       ML0(hi, lo, (*fx)[0], ptr[ 0]); | ||||
|       MLA(hi, lo, (*fx)[1], ptr[14]); | ||||
|       MLA(hi, lo, (*fx)[2], ptr[12]); | ||||
|       MLA(hi, lo, (*fx)[3], ptr[10]); | ||||
|       MLA(hi, lo, (*fx)[4], ptr[ 8]); | ||||
|       MLA(hi, lo, (*fx)[5], ptr[ 6]); | ||||
|       MLA(hi, lo, (*fx)[6], ptr[ 4]); | ||||
|       MLA(hi, lo, (*fx)[7], ptr[ 2]); | ||||
|       MLN(hi, lo); | ||||
| 
 | ||||
|       ptr = *Dptr + pe; | ||||
|       MLA(hi, lo, (*fe)[0], ptr[ 0]); | ||||
|       MLA(hi, lo, (*fe)[1], ptr[14]); | ||||
|       MLA(hi, lo, (*fe)[2], ptr[12]); | ||||
|       MLA(hi, lo, (*fe)[3], ptr[10]); | ||||
|       MLA(hi, lo, (*fe)[4], ptr[ 8]); | ||||
|       MLA(hi, lo, (*fe)[5], ptr[ 6]); | ||||
|       MLA(hi, lo, (*fe)[6], ptr[ 4]); | ||||
|       MLA(hi, lo, (*fe)[7], ptr[ 2]); | ||||
| 
 | ||||
|       raw_sample = SHIFT(MLZ(hi, lo)); | ||||
|       raw_sample = scale(raw_sample); | ||||
|       (*pcm1++) += (short int)raw_sample; | ||||
|       pcm2 = pcm1 + 30; | ||||
| 
 | ||||
|       for (sb = 1; sb < 16; ++sb) | ||||
|       { | ||||
|         ++fe; | ||||
| 	++Dptr; | ||||
| 
 | ||||
| 	/* D[32 - sb][i] == -D[sb][31 - i] */ | ||||
| 
 | ||||
| 	ptr = *Dptr + po; | ||||
| 	ML0(hi, lo, (*fo)[0], ptr[ 0]); | ||||
| 	MLA(hi, lo, (*fo)[1], ptr[14]); | ||||
| 	MLA(hi, lo, (*fo)[2], ptr[12]); | ||||
| 	MLA(hi, lo, (*fo)[3], ptr[10]); | ||||
| 	MLA(hi, lo, (*fo)[4], ptr[ 8]); | ||||
|         MLA(hi, lo, (*fo)[5], ptr[ 6]); | ||||
|         MLA(hi, lo, (*fo)[6], ptr[ 4]); | ||||
|         MLA(hi, lo, (*fo)[7], ptr[ 2]); | ||||
|         MLN(hi, lo); | ||||
| 
 | ||||
|         ptr = *Dptr + pe; | ||||
| 
 | ||||
|         MLA(hi, lo, (*fe)[7], ptr[ 2]); | ||||
|         MLA(hi, lo, (*fe)[6], ptr[ 4]); | ||||
|         MLA(hi, lo, (*fe)[5], ptr[ 6]); | ||||
|         MLA(hi, lo, (*fe)[4], ptr[ 8]); | ||||
|         MLA(hi, lo, (*fe)[3], ptr[10]); | ||||
|         MLA(hi, lo, (*fe)[2], ptr[12]); | ||||
|         MLA(hi, lo, (*fe)[1], ptr[14]); | ||||
|         MLA(hi, lo, (*fe)[0], ptr[ 0]); | ||||
| 
 | ||||
|         raw_sample = SHIFT(MLZ(hi, lo)); | ||||
|         raw_sample = scale(raw_sample); | ||||
|         (*pcm1++) += (short int)raw_sample; | ||||
| 
 | ||||
|         ptr = *Dptr - pe; | ||||
|         ML0(hi, lo, (*fe)[0], ptr[31 - 16]); | ||||
|         MLA(hi, lo, (*fe)[1], ptr[31 - 14]); | ||||
|         MLA(hi, lo, (*fe)[2], ptr[31 - 12]); | ||||
|         MLA(hi, lo, (*fe)[3], ptr[31 - 10]); | ||||
|         MLA(hi, lo, (*fe)[4], ptr[31 -  8]); | ||||
|         MLA(hi, lo, (*fe)[5], ptr[31 -  6]); | ||||
|         MLA(hi, lo, (*fe)[6], ptr[31 -  4]); | ||||
|         MLA(hi, lo, (*fe)[7], ptr[31 -  2]); | ||||
| 
 | ||||
|         ptr = *Dptr - po; | ||||
|         MLA(hi, lo, (*fo)[7], ptr[31 -  2]); | ||||
|         MLA(hi, lo, (*fo)[6], ptr[31 -  4]); | ||||
|         MLA(hi, lo, (*fo)[5], ptr[31 -  6]); | ||||
|         MLA(hi, lo, (*fo)[4], ptr[31 -  8]); | ||||
|         MLA(hi, lo, (*fo)[3], ptr[31 - 10]); | ||||
|         MLA(hi, lo, (*fo)[2], ptr[31 - 12]); | ||||
|         MLA(hi, lo, (*fo)[1], ptr[31 - 14]); | ||||
|         MLA(hi, lo, (*fo)[0], ptr[31 - 16]); | ||||
| 
 | ||||
|         raw_sample = SHIFT(MLZ(hi, lo)); | ||||
|         raw_sample = scale(raw_sample); | ||||
|         (*pcm2--) += (short int)raw_sample; | ||||
| 
 | ||||
|         ++fo; | ||||
|       } | ||||
| 
 | ||||
|       Dptr++; | ||||
| 
 | ||||
|       ptr = *Dptr + po; | ||||
|       ML0(hi, lo, (*fo)[0], ptr[ 0]); | ||||
|       MLA(hi, lo, (*fo)[1], ptr[14]); | ||||
|       MLA(hi, lo, (*fo)[2], ptr[12]); | ||||
|       MLA(hi, lo, (*fo)[3], ptr[10]); | ||||
|       MLA(hi, lo, (*fo)[4], ptr[ 8]); | ||||
|       MLA(hi, lo, (*fo)[5], ptr[ 6]); | ||||
|       MLA(hi, lo, (*fo)[6], ptr[ 4]); | ||||
|       MLA(hi, lo, (*fo)[7], ptr[ 2]); | ||||
| 
 | ||||
|       raw_sample = SHIFT(-MLZ(hi, lo)); | ||||
|       raw_sample = scale(raw_sample); | ||||
|       (*pcm1) += (short int)raw_sample; | ||||
| 
 | ||||
|     }  /* for di canale */ | ||||
| 
 | ||||
|     /* Render di un blocco */     | ||||
|     if(nch < 2) memcpy(short_sample_buff[1], short_sample_buff[0], sizeof(short_sample_buff[0])); | ||||
|     render_sample_block(short_sample_buff, sizeof(short_sample_buff[0])/sizeof(short int)); | ||||
|      | ||||
|     phase = (phase + 1) % 16; | ||||
| 
 | ||||
|   }	/* for di blocco */ | ||||
| } | ||||
| #endif | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	synth->half() | ||||
|  * DESCRIPTION:	perform half frequency PCM synthesis | ||||
|  */ | ||||
| static | ||||
| void synth_half(struct mad_synth *synth, struct mad_frame const *frame, | ||||
| 		unsigned int nch, unsigned int ns) | ||||
| { | ||||
|   unsigned int phase, ch, s, sb, pe, po; | ||||
|   short int *pcm1, *pcm1v, *pcm2v; | ||||
|   mad_fixed_t (*filter)[2][2][16][8]; | ||||
|   mad_fixed_t (*sbsample)[36][32]; | ||||
|   register mad_fixed_t (*fe)[8], (*fx)[8], (*fo)[8]; | ||||
|   register mad_fixed_t const (*Dptr)[32], *ptr ; | ||||
|   register mad_fixed64hi_t hi; | ||||
|   register mad_fixed64lo_t lo; | ||||
|   mad_fixed_t raw_sample; | ||||
|   short int short_sample_buff[2][16]; | ||||
| 
 | ||||
|   phase = synth->phase; | ||||
| 
 | ||||
|   if (nch > 2) | ||||
|    return; | ||||
| 
 | ||||
|   for (s = 0; s < ns; ++s) | ||||
|   { | ||||
|     memset (short_sample_buff, 0x00, sizeof(short_sample_buff)); | ||||
| 
 | ||||
|     for (ch = 0; ch < nch; ++ch) | ||||
|     { | ||||
|       sbsample = &frame->sbsample[ch]; | ||||
|       filter   = &synth->filter[ch]; | ||||
|       pcm1 = pcm1v = short_sample_buff; | ||||
| 
 | ||||
|       dct32((*sbsample)[s], phase >> 1, | ||||
| 	    (*filter)[0][phase & 1], (*filter)[1][phase & 1]); | ||||
| 
 | ||||
|       pe = phase & ~1; | ||||
|       po = ((phase - 1) & 0xf) | 1; | ||||
| 
 | ||||
|       /* calculate 16 samples */ | ||||
| 
 | ||||
|       fe = &(*filter)[0][ phase & 1][0]; | ||||
|       fx = &(*filter)[0][~phase & 1][0]; | ||||
|       fo = &(*filter)[1][~phase & 1][0]; | ||||
| 
 | ||||
|       Dptr = &D[0]; | ||||
| 
 | ||||
|       ptr = *Dptr + po; | ||||
|       ML0(hi, lo, (*fx)[0], ptr[ 0]); | ||||
|       MLA(hi, lo, (*fx)[1], ptr[14]); | ||||
|       MLA(hi, lo, (*fx)[2], ptr[12]); | ||||
|       MLA(hi, lo, (*fx)[3], ptr[10]); | ||||
|       MLA(hi, lo, (*fx)[4], ptr[ 8]); | ||||
|       MLA(hi, lo, (*fx)[5], ptr[ 6]); | ||||
|       MLA(hi, lo, (*fx)[6], ptr[ 4]); | ||||
|       MLA(hi, lo, (*fx)[7], ptr[ 2]); | ||||
|       MLN(hi, lo); | ||||
| 
 | ||||
|       ptr = *Dptr + pe; | ||||
|       MLA(hi, lo, (*fe)[0], ptr[ 0]); | ||||
|       MLA(hi, lo, (*fe)[1], ptr[14]); | ||||
|       MLA(hi, lo, (*fe)[2], ptr[12]); | ||||
|       MLA(hi, lo, (*fe)[3], ptr[10]); | ||||
|       MLA(hi, lo, (*fe)[4], ptr[ 8]); | ||||
|       MLA(hi, lo, (*fe)[5], ptr[ 6]); | ||||
|       MLA(hi, lo, (*fe)[6], ptr[ 4]); | ||||
|       MLA(hi, lo, (*fe)[7], ptr[ 2]); | ||||
| 
 | ||||
|       raw_sample = SHIFT(MLZ(hi, lo)); | ||||
|       raw_sample = scale(raw_sample); | ||||
|       (*pcm1v++) += (short int)raw_sample; | ||||
|       pcm2v = pcm1v + 14; | ||||
| 
 | ||||
|       for (sb = 1; sb < 16; ++sb) | ||||
|       { | ||||
|         ++fe; | ||||
| 	++Dptr; | ||||
| 
 | ||||
| 	/* D[32 - sb][i] == -D[sb][31 - i] */ | ||||
| 
 | ||||
| 	ptr = *Dptr + po; | ||||
|         ML0(hi, lo, (*fo)[0], ptr[ 0]); | ||||
| 	MLA(hi, lo, (*fo)[1], ptr[14]); | ||||
| 	MLA(hi, lo, (*fo)[2], ptr[12]); | ||||
| 	MLA(hi, lo, (*fo)[3], ptr[10]); | ||||
| 	MLA(hi, lo, (*fo)[4], ptr[ 8]); | ||||
| 	MLA(hi, lo, (*fo)[5], ptr[ 6]); | ||||
| 	MLA(hi, lo, (*fo)[6], ptr[ 4]); | ||||
| 	MLA(hi, lo, (*fo)[7], ptr[ 2]); | ||||
| 	MLN(hi, lo); | ||||
| 
 | ||||
| 	ptr = *Dptr + pe; | ||||
| 
 | ||||
|         MLA(hi, lo, (*fe)[7], ptr[ 2]); | ||||
| 	MLA(hi, lo, (*fe)[6], ptr[ 4]); | ||||
| 	MLA(hi, lo, (*fe)[5], ptr[ 6]); | ||||
| 	MLA(hi, lo, (*fe)[4], ptr[ 8]); | ||||
| 	MLA(hi, lo, (*fe)[3], ptr[10]); | ||||
| 	MLA(hi, lo, (*fe)[2], ptr[12]); | ||||
| 	MLA(hi, lo, (*fe)[1], ptr[14]); | ||||
| 	MLA(hi, lo, (*fe)[0], ptr[ 0]); | ||||
| 
 | ||||
| 	raw_sample = SHIFT(MLZ(hi, lo)); | ||||
| 	raw_sample = scale(raw_sample); | ||||
| 	(*pcm1v++) += (short int)raw_sample; | ||||
| 
 | ||||
| 	ptr = *Dptr - pe; | ||||
| 	ML0(hi, lo, (*fe)[0], ptr[31 - 16]); | ||||
| 	MLA(hi, lo, (*fe)[1], ptr[31 - 14]); | ||||
| 	MLA(hi, lo, (*fe)[2], ptr[31 - 12]); | ||||
| 	MLA(hi, lo, (*fe)[3], ptr[31 - 10]); | ||||
| 	MLA(hi, lo, (*fe)[4], ptr[31 -  8]); | ||||
| 	MLA(hi, lo, (*fe)[5], ptr[31 -  6]); | ||||
| 	MLA(hi, lo, (*fe)[6], ptr[31 -  4]); | ||||
| 	MLA(hi, lo, (*fe)[7], ptr[31 -  2]); | ||||
| 
 | ||||
| 	ptr = *Dptr - po; | ||||
| 	MLA(hi, lo, (*fo)[7], ptr[31 -  2]); | ||||
| 	MLA(hi, lo, (*fo)[6], ptr[31 -  4]); | ||||
| 	MLA(hi, lo, (*fo)[5], ptr[31 -  6]); | ||||
| 	MLA(hi, lo, (*fo)[4], ptr[31 -  8]); | ||||
| 	MLA(hi, lo, (*fo)[3], ptr[31 - 10]); | ||||
| 	MLA(hi, lo, (*fo)[2], ptr[31 - 12]); | ||||
| 	MLA(hi, lo, (*fo)[1], ptr[31 - 14]); | ||||
| 	MLA(hi, lo, (*fo)[0], ptr[31 - 16]); | ||||
| 
 | ||||
| 	raw_sample = SHIFT(MLZ(hi, lo)); | ||||
| 	raw_sample = scale(raw_sample); | ||||
| 	(*pcm2v--) += (short int)raw_sample; | ||||
| 
 | ||||
|         ++fo; | ||||
|       } | ||||
| 
 | ||||
|       Dptr++; | ||||
| 
 | ||||
|       ptr = *Dptr + po; | ||||
|       ML0(hi, lo, (*fo)[0], ptr[ 0]); | ||||
|       MLA(hi, lo, (*fo)[1], ptr[14]); | ||||
|       MLA(hi, lo, (*fo)[2], ptr[12]); | ||||
|       MLA(hi, lo, (*fo)[3], ptr[10]); | ||||
|       MLA(hi, lo, (*fo)[4], ptr[ 8]); | ||||
|       MLA(hi, lo, (*fo)[5], ptr[ 6]); | ||||
|       MLA(hi, lo, (*fo)[6], ptr[ 4]); | ||||
|       MLA(hi, lo, (*fo)[7], ptr[ 2]); | ||||
| 
 | ||||
|       raw_sample = SHIFT(-MLZ(hi, lo)); | ||||
|       raw_sample = scale(raw_sample); | ||||
|       (*pcm1v) += (short int)raw_sample; | ||||
| 
 | ||||
|     }  /* for di canale */ | ||||
| 
 | ||||
|     /* Render di un blocco */ | ||||
|     if(nch < 2) memcpy(short_sample_buff[1], short_sample_buff[0], sizeof(short_sample_buff[0])); | ||||
|     render_sample_block(short_sample_buff, sizeof(short_sample_buff[0])/sizeof(short int)); | ||||
| 
 | ||||
|     pcm1 = pcm1v + 8; | ||||
|     phase = (phase + 1) % 16; | ||||
| 
 | ||||
|   }	/* for di blocco */ | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	synth->frame() | ||||
|  * DESCRIPTION:	perform PCM synthesis of frame subband samples | ||||
|  */ | ||||
| void mad_synth_frame(struct mad_synth *synth, struct mad_frame const *frame) | ||||
| { | ||||
|   unsigned int nch, ns; | ||||
|   void (*synth_frame)(struct mad_synth *, struct mad_frame const *, | ||||
| 		      unsigned int, unsigned int); | ||||
| 
 | ||||
|   nch = MAD_NCHANNELS(&frame->header); | ||||
|   ns  = MAD_NSBSAMPLES(&frame->header); | ||||
| 
 | ||||
|   synth->pcm.samplerate = frame->header.samplerate; | ||||
|   synth->pcm.channels   = nch; | ||||
| //  synth->pcm.length     = 32 * ns;
 | ||||
|   synth->pcm.length     = 64 * ns; | ||||
| 
 | ||||
|   synth_frame = synth_full; | ||||
| 
 | ||||
|   if (frame->options & MAD_OPTION_HALFSAMPLERATE) { | ||||
|     synth->pcm.samplerate /= 2; | ||||
|     synth->pcm.length     /= 2; | ||||
|     synth_frame = synth_half; | ||||
|   } | ||||
|   set_dac_sample_rate(synth->pcm.samplerate, nch); | ||||
| 
 | ||||
|   synth_frame(synth, frame, nch, ns); | ||||
|   synth->phase = (synth->phase + ns) % 16; | ||||
| } | ||||
							
								
								
									
										929
									
								
								project/src/mad/synth_mono.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										929
									
								
								project/src/mad/synth_mono.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,929 @@ | |||
| /*
 | ||||
|  * libmad - MPEG audio decoder library | ||||
|  * Copyright (C) 2000-2004 Underbit Technologies, Inc. | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|  * the Free Software Foundation; either version 2 of the License, or | ||||
|  * (at your option) any later version. | ||||
|  * | ||||
|  * This program 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.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  * | ||||
|  * You should have received a copy of the GNU General Public License | ||||
|  * along with this program; if not, write to the Free Software | ||||
|  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA | ||||
|  * | ||||
|  * $Id: synth.c,v 1.25 2004/01/23 09:41:33 rob Exp $ | ||||
|  */ | ||||
| 
 | ||||
| # ifdef HAVE_CONFIG_H | ||||
| #  include "config.h" | ||||
| # endif | ||||
| 
 | ||||
| # include "global.h" | ||||
| 
 | ||||
| # include "fixed.h" | ||||
| # include "frame.h" | ||||
| # include "synth.h" | ||||
| # include "string.h" | ||||
| 
 | ||||
| 
 | ||||
| /*
 | ||||
|  * The following utility routine performs simple rounding, clipping, and | ||||
|  * scaling of MAD's high-resolution samples down to 16 bits. It does not | ||||
|  * perform any dithering or noise shaping, which would be recommended to | ||||
|  * obtain any exceptional audio quality. It is therefore not recommended to | ||||
|  * use this routine if high-quality output is desired. | ||||
|  */ | ||||
| 
 | ||||
| static inline | ||||
| signed short scale(mad_fixed_t sample) | ||||
| { | ||||
| 
 | ||||
|   /* round */ | ||||
|   sample += (1L << (MAD_F_FRACBITS - 16)); | ||||
| 
 | ||||
|   /* clip */ | ||||
|   if (sample >= MAD_F_ONE) sample = MAD_F_ONE - 1; | ||||
|   else if (sample < -MAD_F_ONE) sample = -MAD_F_ONE; | ||||
| 
 | ||||
|   /* quantize */ | ||||
|   //The original nxp code had
 | ||||
|   //return sample >> (MAD_F_FRACBITS + 1 - 16);
 | ||||
|   //but somehow that clipped and distorted on loud sounds...
 | ||||
|   //This seems to be OK:
 | ||||
|   return sample >> (MAD_F_FRACBITS + 2 - 16); | ||||
| } | ||||
| /*
 | ||||
|  * NAME:	synth->init() | ||||
|  * DESCRIPTION:	initialize synth struct | ||||
|  */ | ||||
| void ICACHE_FLASH_ATTR mad_synth_init(struct mad_synth *synth) | ||||
| { | ||||
|   mad_synth_mute(synth); | ||||
| 
 | ||||
|   synth->phase = 0; | ||||
| 
 | ||||
|   synth->pcm.samplerate = 0; | ||||
|   synth->pcm.channels   = 0; | ||||
|   synth->pcm.length     = 0; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	synth->mute() | ||||
|  * DESCRIPTION:	zero all polyphase filterbank values, resetting synthesis | ||||
|  */ | ||||
| void ICACHE_FLASH_ATTR mad_synth_mute(struct mad_synth *synth) | ||||
| { | ||||
|   unsigned int ch, s, v; | ||||
| 
 | ||||
|   for (ch = 0; ch < 2; ++ch) { | ||||
|     for (s = 0; s < 16; ++s) { | ||||
|       for (v = 0; v < 8; ++v) { | ||||
| 	synth->filter[ch][0][0][s][v] = synth->filter[ch][0][1][s][v] = | ||||
| 	synth->filter[ch][1][0][s][v] = synth->filter[ch][1][1][s][v] = 0; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * An optional optimization called here the Subband Synthesis Optimization | ||||
|  * (SSO) improves the performance of subband synthesis at the expense of | ||||
|  * accuracy. | ||||
|  * | ||||
|  * The idea is to simplify 32x32->64-bit multiplication to 32x32->32 such | ||||
|  * that extra scaling and rounding are not necessary. This often allows the | ||||
|  * compiler to use faster 32-bit multiply-accumulate instructions instead of | ||||
|  * explicit 64-bit multiply, shift, and add instructions. | ||||
|  * | ||||
|  * SSO works like this: a full 32x32->64-bit multiply of two mad_fixed_t | ||||
|  * values requires the result to be right-shifted 28 bits to be properly | ||||
|  * scaled to the same fixed-point format. Right shifts can be applied at any | ||||
|  * time to either operand or to the result, so the optimization involves | ||||
|  * careful placement of these shifts to minimize the loss of accuracy. | ||||
|  * | ||||
|  * First, a 14-bit shift is applied with rounding at compile-time to the D[] | ||||
|  * table of coefficients for the subband synthesis window. This only loses 2 | ||||
|  * bits of accuracy because the lower 12 bits are always zero. A second | ||||
|  * 12-bit shift occurs after the DCT calculation. This loses 12 bits of | ||||
|  * accuracy. Finally, a third 2-bit shift occurs just before the sample is | ||||
|  * saved in the PCM buffer. 14 + 12 + 2 == 28 bits. | ||||
|  */ | ||||
| 
 | ||||
| /* FPM_DEFAULT without OPT_SSO will actually lose accuracy and performance */ | ||||
| 
 | ||||
| # if defined(FPM_DEFAULT) && !defined(OPT_SSO) | ||||
| #  define OPT_SSO | ||||
| # endif | ||||
| 
 | ||||
| /* second SSO shift, with rounding */ | ||||
| 
 | ||||
| # if defined(OPT_SSO) | ||||
| #  define SHIFT(x)  (((x) + (1L << 11)) >> 12) | ||||
| # else | ||||
| #  define SHIFT(x)  (x) | ||||
| # endif | ||||
| 
 | ||||
| /* possible DCT speed optimization */ | ||||
| 
 | ||||
| # if defined(OPT_SPEED) && defined(MAD_F_MLX) | ||||
| #  define OPT_DCTO | ||||
| #  define MUL(x, y)  \ | ||||
|     ({ mad_fixed64hi_t hi;  \ | ||||
|        mad_fixed64lo_t lo;  \ | ||||
|        MAD_F_MLX(hi, lo, (x), (y));  \ | ||||
|        hi << (32 - MAD_F_SCALEBITS - 3);  \ | ||||
|     }) | ||||
| # else | ||||
| #  undef OPT_DCTO | ||||
| #  define MUL(x, y)  mad_f_mul((x), (y)) | ||||
| # endif | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	dct32() | ||||
|  * DESCRIPTION:	perform fast in[32]->out[32] DCT | ||||
|  */ | ||||
| 
 | ||||
| static | ||||
| void ICACHE_FLASH_ATTR dct32(mad_fixed_t const in[32], unsigned int slot, | ||||
| 	   mad_fixed_t lo[16][8], mad_fixed_t hi[16][8]) | ||||
| { | ||||
|   mad_fixed_t t0,   t1,   t2,   t3,   t4,   t5,   t6,   t7; | ||||
|   mad_fixed_t t8,   t9,   t10,  t11,  t12,  t13,  t14,  t15; | ||||
|   mad_fixed_t t16,  t17,  t18,  t19,  t20,  t21,  t22,  t23; | ||||
|   mad_fixed_t t24,  t25,  t26,  t27,  t28,  t29,  t30,  t31; | ||||
|   mad_fixed_t t32,  t33,  t34,  t35,  t36,  t37,  t38,  t39; | ||||
|   mad_fixed_t t40,  t41,  t42,  t43,  t44,  t45,  t46,  t47; | ||||
|   mad_fixed_t t48,  t49,  t50,  t51,  t52,  t53,  t54,  t55; | ||||
|   mad_fixed_t t56,  t57,  t58,  t59,  t60,  t61,  t62,  t63; | ||||
|   mad_fixed_t t64,  t65,  t66,  t67,  t68,  t69,  t70,  t71; | ||||
|   mad_fixed_t t72,  t73,  t74,  t75,  t76,  t77,  t78,  t79; | ||||
|   mad_fixed_t t80,  t81,  t82,  t83,  t84,  t85,  t86,  t87; | ||||
|   mad_fixed_t t88,  t89,  t90,  t91,  t92,  t93,  t94,  t95; | ||||
|   mad_fixed_t t96,  t97,  t98,  t99,  t100, t101, t102, t103; | ||||
|   mad_fixed_t t104, t105, t106, t107, t108, t109, t110, t111; | ||||
|   mad_fixed_t t112, t113, t114, t115, t116, t117, t118, t119; | ||||
|   mad_fixed_t t120, t121, t122, t123, t124, t125, t126, t127; | ||||
|   mad_fixed_t t128, t129, t130, t131, t132, t133, t134, t135; | ||||
|   mad_fixed_t t136, t137, t138, t139, t140, t141, t142, t143; | ||||
|   mad_fixed_t t144, t145, t146, t147, t148, t149, t150, t151; | ||||
|   mad_fixed_t t152, t153, t154, t155, t156, t157, t158, t159; | ||||
|   mad_fixed_t t160, t161, t162, t163, t164, t165, t166, t167; | ||||
|   mad_fixed_t t168, t169, t170, t171, t172, t173, t174, t175; | ||||
|   mad_fixed_t t176; | ||||
| 
 | ||||
|   /* costab[i] = cos(PI / (2 * 32) * i) */ | ||||
| 
 | ||||
| # if defined(OPT_DCTO) | ||||
| #  define costab1	MAD_F(0x7fd8878e) | ||||
| #  define costab2	MAD_F(0x7f62368f) | ||||
| #  define costab3	MAD_F(0x7e9d55fc) | ||||
| #  define costab4	MAD_F(0x7d8a5f40) | ||||
| #  define costab5	MAD_F(0x7c29fbee) | ||||
| #  define costab6	MAD_F(0x7a7d055b) | ||||
| #  define costab7	MAD_F(0x78848414) | ||||
| #  define costab8	MAD_F(0x7641af3d) | ||||
| #  define costab9	MAD_F(0x73b5ebd1) | ||||
| #  define costab10	MAD_F(0x70e2cbc6) | ||||
| #  define costab11	MAD_F(0x6dca0d14) | ||||
| #  define costab12	MAD_F(0x6a6d98a4) | ||||
| #  define costab13	MAD_F(0x66cf8120) | ||||
| #  define costab14	MAD_F(0x62f201ac) | ||||
| #  define costab15	MAD_F(0x5ed77c8a) | ||||
| #  define costab16	MAD_F(0x5a82799a) | ||||
| #  define costab17	MAD_F(0x55f5a4d2) | ||||
| #  define costab18	MAD_F(0x5133cc94) | ||||
| #  define costab19	MAD_F(0x4c3fdff4) | ||||
| #  define costab20	MAD_F(0x471cece7) | ||||
| #  define costab21	MAD_F(0x41ce1e65) | ||||
| #  define costab22	MAD_F(0x3c56ba70) | ||||
| #  define costab23	MAD_F(0x36ba2014) | ||||
| #  define costab24	MAD_F(0x30fbc54d) | ||||
| #  define costab25	MAD_F(0x2b1f34eb) | ||||
| #  define costab26	MAD_F(0x25280c5e) | ||||
| #  define costab27	MAD_F(0x1f19f97b) | ||||
| #  define costab28	MAD_F(0x18f8b83c) | ||||
| #  define costab29	MAD_F(0x12c8106f) | ||||
| #  define costab30	MAD_F(0x0c8bd35e) | ||||
| #  define costab31	MAD_F(0x0647d97c) | ||||
| # else | ||||
| #  define costab1	MAD_F(0x0ffb10f2)  /* 0.998795456 */ | ||||
| #  define costab2	MAD_F(0x0fec46d2)  /* 0.995184727 */ | ||||
| #  define costab3	MAD_F(0x0fd3aac0)  /* 0.989176510 */ | ||||
| #  define costab4	MAD_F(0x0fb14be8)  /* 0.980785280 */ | ||||
| #  define costab5	MAD_F(0x0f853f7e)  /* 0.970031253 */ | ||||
| #  define costab6	MAD_F(0x0f4fa0ab)  /* 0.956940336 */ | ||||
| #  define costab7	MAD_F(0x0f109082)  /* 0.941544065 */ | ||||
| #  define costab8	MAD_F(0x0ec835e8)  /* 0.923879533 */ | ||||
| #  define costab9	MAD_F(0x0e76bd7a)  /* 0.903989293 */ | ||||
| #  define costab10	MAD_F(0x0e1c5979)  /* 0.881921264 */ | ||||
| #  define costab11	MAD_F(0x0db941a3)  /* 0.857728610 */ | ||||
| #  define costab12	MAD_F(0x0d4db315)  /* 0.831469612 */ | ||||
| #  define costab13	MAD_F(0x0cd9f024)  /* 0.803207531 */ | ||||
| #  define costab14	MAD_F(0x0c5e4036)  /* 0.773010453 */ | ||||
| #  define costab15	MAD_F(0x0bdaef91)  /* 0.740951125 */ | ||||
| #  define costab16	MAD_F(0x0b504f33)  /* 0.707106781 */ | ||||
| #  define costab17	MAD_F(0x0abeb49a)  /* 0.671558955 */ | ||||
| #  define costab18	MAD_F(0x0a267993)  /* 0.634393284 */ | ||||
| #  define costab19	MAD_F(0x0987fbfe)  /* 0.595699304 */ | ||||
| #  define costab20	MAD_F(0x08e39d9d)  /* 0.555570233 */ | ||||
| #  define costab21	MAD_F(0x0839c3cd)  /* 0.514102744 */ | ||||
| #  define costab22	MAD_F(0x078ad74e)  /* 0.471396737 */ | ||||
| #  define costab23	MAD_F(0x06d74402)  /* 0.427555093 */ | ||||
| #  define costab24	MAD_F(0x061f78aa)  /* 0.382683432 */ | ||||
| #  define costab25	MAD_F(0x0563e69d)  /* 0.336889853 */ | ||||
| #  define costab26	MAD_F(0x04a5018c)  /* 0.290284677 */ | ||||
| #  define costab27	MAD_F(0x03e33f2f)  /* 0.242980180 */ | ||||
| #  define costab28	MAD_F(0x031f1708)  /* 0.195090322 */ | ||||
| #  define costab29	MAD_F(0x0259020e)  /* 0.146730474 */ | ||||
| #  define costab30	MAD_F(0x01917a6c)  /* 0.098017140 */ | ||||
| #  define costab31	MAD_F(0x00c8fb30)  /* 0.049067674 */ | ||||
| # endif | ||||
| 
 | ||||
|   t0   = in[0]  + in[31];  t16  = MUL(in[0]  - in[31], costab1); | ||||
|   t1   = in[15] + in[16];  t17  = MUL(in[15] - in[16], costab31); | ||||
| 
 | ||||
|   t41  = t16 + t17; | ||||
|   t59  = MUL(t16 - t17, costab2); | ||||
|   t33  = t0  + t1; | ||||
|   t50  = MUL(t0  - t1,  costab2); | ||||
| 
 | ||||
|   t2   = in[7]  + in[24];  t18  = MUL(in[7]  - in[24], costab15); | ||||
|   t3   = in[8]  + in[23];  t19  = MUL(in[8]  - in[23], costab17); | ||||
| 
 | ||||
|   t42  = t18 + t19; | ||||
|   t60  = MUL(t18 - t19, costab30); | ||||
|   t34  = t2  + t3; | ||||
|   t51  = MUL(t2  - t3,  costab30); | ||||
| 
 | ||||
|   t4   = in[3]  + in[28];  t20  = MUL(in[3]  - in[28], costab7); | ||||
|   t5   = in[12] + in[19];  t21  = MUL(in[12] - in[19], costab25); | ||||
| 
 | ||||
|   t43  = t20 + t21; | ||||
|   t61  = MUL(t20 - t21, costab14); | ||||
|   t35  = t4  + t5; | ||||
|   t52  = MUL(t4  - t5,  costab14); | ||||
| 
 | ||||
|   t6   = in[4]  + in[27];  t22  = MUL(in[4]  - in[27], costab9); | ||||
|   t7   = in[11] + in[20];  t23  = MUL(in[11] - in[20], costab23); | ||||
| 
 | ||||
|   t44  = t22 + t23; | ||||
|   t62  = MUL(t22 - t23, costab18); | ||||
|   t36  = t6  + t7; | ||||
|   t53  = MUL(t6  - t7,  costab18); | ||||
| 
 | ||||
|   t8   = in[1]  + in[30];  t24  = MUL(in[1]  - in[30], costab3); | ||||
|   t9   = in[14] + in[17];  t25  = MUL(in[14] - in[17], costab29); | ||||
| 
 | ||||
|   t45  = t24 + t25; | ||||
|   t63  = MUL(t24 - t25, costab6); | ||||
|   t37  = t8  + t9; | ||||
|   t54  = MUL(t8  - t9,  costab6); | ||||
| 
 | ||||
|   t10  = in[6]  + in[25];  t26  = MUL(in[6]  - in[25], costab13); | ||||
|   t11  = in[9]  + in[22];  t27  = MUL(in[9]  - in[22], costab19); | ||||
| 
 | ||||
|   t46  = t26 + t27; | ||||
|   t64  = MUL(t26 - t27, costab26); | ||||
|   t38  = t10 + t11; | ||||
|   t55  = MUL(t10 - t11, costab26); | ||||
| 
 | ||||
|   t12  = in[2]  + in[29];  t28  = MUL(in[2]  - in[29], costab5); | ||||
|   t13  = in[13] + in[18];  t29  = MUL(in[13] - in[18], costab27); | ||||
| 
 | ||||
|   t47  = t28 + t29; | ||||
|   t65  = MUL(t28 - t29, costab10); | ||||
|   t39  = t12 + t13; | ||||
|   t56  = MUL(t12 - t13, costab10); | ||||
| 
 | ||||
|   t14  = in[5]  + in[26];  t30  = MUL(in[5]  - in[26], costab11); | ||||
|   t15  = in[10] + in[21];  t31  = MUL(in[10] - in[21], costab21); | ||||
| 
 | ||||
|   t48  = t30 + t31; | ||||
|   t66  = MUL(t30 - t31, costab22); | ||||
|   t40  = t14 + t15; | ||||
|   t57  = MUL(t14 - t15, costab22); | ||||
| 
 | ||||
|   t69  = t33 + t34;  t89  = MUL(t33 - t34, costab4); | ||||
|   t70  = t35 + t36;  t90  = MUL(t35 - t36, costab28); | ||||
|   t71  = t37 + t38;  t91  = MUL(t37 - t38, costab12); | ||||
|   t72  = t39 + t40;  t92  = MUL(t39 - t40, costab20); | ||||
|   t73  = t41 + t42;  t94  = MUL(t41 - t42, costab4); | ||||
|   t74  = t43 + t44;  t95  = MUL(t43 - t44, costab28); | ||||
|   t75  = t45 + t46;  t96  = MUL(t45 - t46, costab12); | ||||
|   t76  = t47 + t48;  t97  = MUL(t47 - t48, costab20); | ||||
| 
 | ||||
|   t78  = t50 + t51;  t100 = MUL(t50 - t51, costab4); | ||||
|   t79  = t52 + t53;  t101 = MUL(t52 - t53, costab28); | ||||
|   t80  = t54 + t55;  t102 = MUL(t54 - t55, costab12); | ||||
|   t81  = t56 + t57;  t103 = MUL(t56 - t57, costab20); | ||||
| 
 | ||||
|   t83  = t59 + t60;  t106 = MUL(t59 - t60, costab4); | ||||
|   t84  = t61 + t62;  t107 = MUL(t61 - t62, costab28); | ||||
|   t85  = t63 + t64;  t108 = MUL(t63 - t64, costab12); | ||||
|   t86  = t65 + t66;  t109 = MUL(t65 - t66, costab20); | ||||
| 
 | ||||
|   t113 = t69  + t70; | ||||
|   t114 = t71  + t72; | ||||
| 
 | ||||
|   /*  0 */ hi[15][slot] = SHIFT(t113 + t114); | ||||
|   /* 16 */ lo[ 0][slot] = SHIFT(MUL(t113 - t114, costab16)); | ||||
| 
 | ||||
|   t115 = t73  + t74; | ||||
|   t116 = t75  + t76; | ||||
| 
 | ||||
|   t32  = t115 + t116; | ||||
| 
 | ||||
|   /*  1 */ hi[14][slot] = SHIFT(t32); | ||||
| 
 | ||||
|   t118 = t78  + t79; | ||||
|   t119 = t80  + t81; | ||||
| 
 | ||||
|   t58  = t118 + t119; | ||||
| 
 | ||||
|   /*  2 */ hi[13][slot] = SHIFT(t58); | ||||
| 
 | ||||
|   t121 = t83  + t84; | ||||
|   t122 = t85  + t86; | ||||
| 
 | ||||
|   t67  = t121 + t122; | ||||
| 
 | ||||
|   t49  = (t67 * 2) - t32; | ||||
| 
 | ||||
|   /*  3 */ hi[12][slot] = SHIFT(t49); | ||||
| 
 | ||||
|   t125 = t89  + t90; | ||||
|   t126 = t91  + t92; | ||||
| 
 | ||||
|   t93  = t125 + t126; | ||||
| 
 | ||||
|   /*  4 */ hi[11][slot] = SHIFT(t93); | ||||
| 
 | ||||
|   t128 = t94  + t95; | ||||
|   t129 = t96  + t97; | ||||
| 
 | ||||
|   t98  = t128 + t129; | ||||
| 
 | ||||
|   t68  = (t98 * 2) - t49; | ||||
| 
 | ||||
|   /*  5 */ hi[10][slot] = SHIFT(t68); | ||||
| 
 | ||||
|   t132 = t100 + t101; | ||||
|   t133 = t102 + t103; | ||||
| 
 | ||||
|   t104 = t132 + t133; | ||||
| 
 | ||||
|   t82  = (t104 * 2) - t58; | ||||
| 
 | ||||
|   /*  6 */ hi[ 9][slot] = SHIFT(t82); | ||||
| 
 | ||||
|   t136 = t106 + t107; | ||||
|   t137 = t108 + t109; | ||||
| 
 | ||||
|   t110 = t136 + t137; | ||||
| 
 | ||||
|   t87  = (t110 * 2) - t67; | ||||
| 
 | ||||
|   t77  = (t87 * 2) - t68; | ||||
| 
 | ||||
|   /*  7 */ hi[ 8][slot] = SHIFT(t77); | ||||
| 
 | ||||
|   t141 = MUL(t69 - t70, costab8); | ||||
|   t142 = MUL(t71 - t72, costab24); | ||||
|   t143 = t141 + t142; | ||||
| 
 | ||||
|   /*  8 */ hi[ 7][slot] = SHIFT(t143); | ||||
|   /* 24 */ lo[ 8][slot] = | ||||
| 	     SHIFT((MUL(t141 - t142, costab16) * 2) - t143); | ||||
| 
 | ||||
|   t144 = MUL(t73 - t74, costab8); | ||||
|   t145 = MUL(t75 - t76, costab24); | ||||
|   t146 = t144 + t145; | ||||
| 
 | ||||
|   t88  = (t146 * 2) - t77; | ||||
| 
 | ||||
|   /*  9 */ hi[ 6][slot] = SHIFT(t88); | ||||
| 
 | ||||
|   t148 = MUL(t78 - t79, costab8); | ||||
|   t149 = MUL(t80 - t81, costab24); | ||||
|   t150 = t148 + t149; | ||||
| 
 | ||||
|   t105 = (t150 * 2) - t82; | ||||
| 
 | ||||
|   /* 10 */ hi[ 5][slot] = SHIFT(t105); | ||||
| 
 | ||||
|   t152 = MUL(t83 - t84, costab8); | ||||
|   t153 = MUL(t85 - t86, costab24); | ||||
|   t154 = t152 + t153; | ||||
| 
 | ||||
|   t111 = (t154 * 2) - t87; | ||||
| 
 | ||||
|   t99  = (t111 * 2) - t88; | ||||
| 
 | ||||
|   /* 11 */ hi[ 4][slot] = SHIFT(t99); | ||||
| 
 | ||||
|   t157 = MUL(t89 - t90, costab8); | ||||
|   t158 = MUL(t91 - t92, costab24); | ||||
|   t159 = t157 + t158; | ||||
| 
 | ||||
|   t127 = (t159 * 2) - t93; | ||||
| 
 | ||||
|   /* 12 */ hi[ 3][slot] = SHIFT(t127); | ||||
| 
 | ||||
|   t160 = (MUL(t125 - t126, costab16) * 2) - t127; | ||||
| 
 | ||||
|   /* 20 */ lo[ 4][slot] = SHIFT(t160); | ||||
|   /* 28 */ lo[12][slot] = | ||||
| 	     SHIFT((((MUL(t157 - t158, costab16) * 2) - t159) * 2) - t160); | ||||
| 
 | ||||
|   t161 = MUL(t94 - t95, costab8); | ||||
|   t162 = MUL(t96 - t97, costab24); | ||||
|   t163 = t161 + t162; | ||||
| 
 | ||||
|   t130 = (t163 * 2) - t98; | ||||
| 
 | ||||
|   t112 = (t130 * 2) - t99; | ||||
| 
 | ||||
|   /* 13 */ hi[ 2][slot] = SHIFT(t112); | ||||
| 
 | ||||
|   t164 = (MUL(t128 - t129, costab16) * 2) - t130; | ||||
| 
 | ||||
|   t166 = MUL(t100 - t101, costab8); | ||||
|   t167 = MUL(t102 - t103, costab24); | ||||
|   t168 = t166 + t167; | ||||
| 
 | ||||
|   t134 = (t168 * 2) - t104; | ||||
| 
 | ||||
|   t120 = (t134 * 2) - t105; | ||||
| 
 | ||||
|   /* 14 */ hi[ 1][slot] = SHIFT(t120); | ||||
| 
 | ||||
|   t135 = (MUL(t118 - t119, costab16) * 2) - t120; | ||||
| 
 | ||||
|   /* 18 */ lo[ 2][slot] = SHIFT(t135); | ||||
| 
 | ||||
|   t169 = (MUL(t132 - t133, costab16) * 2) - t134; | ||||
| 
 | ||||
|   t151 = (t169 * 2) - t135; | ||||
| 
 | ||||
|   /* 22 */ lo[ 6][slot] = SHIFT(t151); | ||||
| 
 | ||||
|   t170 = (((MUL(t148 - t149, costab16) * 2) - t150) * 2) - t151; | ||||
| 
 | ||||
|   /* 26 */ lo[10][slot] = SHIFT(t170); | ||||
|   /* 30 */ lo[14][slot] = | ||||
| 	     SHIFT((((((MUL(t166 - t167, costab16) * 2) - | ||||
| 		       t168) * 2) - t169) * 2) - t170); | ||||
| 
 | ||||
|   t171 = MUL(t106 - t107, costab8); | ||||
|   t172 = MUL(t108 - t109, costab24); | ||||
|   t173 = t171 + t172; | ||||
| 
 | ||||
|   t138 = (t173 * 2) - t110; | ||||
| 
 | ||||
|   t123 = (t138 * 2) - t111; | ||||
| 
 | ||||
|   t139 = (MUL(t121 - t122, costab16) * 2) - t123; | ||||
| 
 | ||||
|   t117 = (t123 * 2) - t112; | ||||
| 
 | ||||
|   /* 15 */ hi[ 0][slot] = SHIFT(t117); | ||||
| 
 | ||||
|   t124 = (MUL(t115 - t116, costab16) * 2) - t117; | ||||
| 
 | ||||
|   /* 17 */ lo[ 1][slot] = SHIFT(t124); | ||||
| 
 | ||||
|   t131 = (t139 * 2) - t124; | ||||
| 
 | ||||
|   /* 19 */ lo[ 3][slot] = SHIFT(t131); | ||||
| 
 | ||||
|   t140 = (t164 * 2) - t131; | ||||
| 
 | ||||
|   /* 21 */ lo[ 5][slot] = SHIFT(t140); | ||||
| 
 | ||||
|   t174 = (MUL(t136 - t137, costab16) * 2) - t138; | ||||
| 
 | ||||
|   t155 = (t174 * 2) - t139; | ||||
| 
 | ||||
|   t147 = (t155 * 2) - t140; | ||||
| 
 | ||||
|   /* 23 */ lo[ 7][slot] = SHIFT(t147); | ||||
| 
 | ||||
|   t156 = (((MUL(t144 - t145, costab16) * 2) - t146) * 2) - t147; | ||||
| 
 | ||||
|   /* 25 */ lo[ 9][slot] = SHIFT(t156); | ||||
| 
 | ||||
|   t175 = (((MUL(t152 - t153, costab16) * 2) - t154) * 2) - t155; | ||||
| 
 | ||||
|   t165 = (t175 * 2) - t156; | ||||
| 
 | ||||
|   /* 27 */ lo[11][slot] = SHIFT(t165); | ||||
| 
 | ||||
|   t176 = (((((MUL(t161 - t162, costab16) * 2) - | ||||
| 	     t163) * 2) - t164) * 2) - t165; | ||||
| 
 | ||||
|   /* 29 */ lo[13][slot] = SHIFT(t176); | ||||
|   /* 31 */ lo[15][slot] = | ||||
| 	     SHIFT((((((((MUL(t171 - t172, costab16) * 2) - | ||||
| 			 t173) * 2) - t174) * 2) - t175) * 2) - t176); | ||||
| 
 | ||||
|   /*
 | ||||
|    * Totals: | ||||
|    *  80 multiplies | ||||
|    *  80 additions | ||||
|    * 119 subtractions | ||||
|    *  49 shifts (not counting SSO) | ||||
|    */ | ||||
| } | ||||
| 
 | ||||
| # undef MUL | ||||
| # undef SHIFT | ||||
| 
 | ||||
| /* third SSO shift and/or D[] optimization preshift */ | ||||
| 
 | ||||
| # if defined(OPT_SSO) | ||||
| #  if MAD_F_FRACBITS != 28 | ||||
| #   error "MAD_F_FRACBITS must be 28 to use OPT_SSO" | ||||
| #  endif | ||||
| #  define ML0(hi, lo, x, y)	((lo)  = (x) * (y)) | ||||
| #  define MLA(hi, lo, x, y)	((lo) += (x) * (y)) | ||||
| #  define MLN(hi, lo)		((lo)  = -(lo)) | ||||
| #  define MLZ(hi, lo)		((void) (hi), (mad_fixed_t) (lo)) | ||||
| #  define SHIFT(x)		((x) >> 2) | ||||
| #  define PRESHIFT(x)		((MAD_F(x) + (1L << 13)) >> 14) | ||||
| # else | ||||
| #  define ML0(hi, lo, x, y)	MAD_F_ML0((hi), (lo), (x), (y)) | ||||
| #  define MLA(hi, lo, x, y)	MAD_F_MLA((hi), (lo), (x), (y)) | ||||
| #  define MLN(hi, lo)		MAD_F_MLN((hi), (lo)) | ||||
| #  define MLZ(hi, lo)		MAD_F_MLZ((hi), (lo)) | ||||
| #  define SHIFT(x)		(x) | ||||
| #  if defined(MAD_F_SCALEBITS) | ||||
| #   undef  MAD_F_SCALEBITS | ||||
| #   define MAD_F_SCALEBITS	(MAD_F_FRACBITS - 12) | ||||
| #   define PRESHIFT(x)		(MAD_F(x) >> 12) | ||||
| #  else | ||||
| #   define PRESHIFT(x)		MAD_F(x) | ||||
| #  endif | ||||
| # endif | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| static | ||||
| mad_fixed_t ICACHE_RODATA_ATTR const D[17][32] = { | ||||
| # include "D.dat" | ||||
| }; | ||||
| 
 | ||||
| # if defined(ASO_SYNTH) | ||||
| void ICACHE_FLASH_ATTR synth_full(struct mad_synth *, struct mad_frame const *, | ||||
| 		unsigned int, unsigned int); | ||||
| # else | ||||
| /*
 | ||||
|  * NAME:	synth->full() | ||||
|  * DESCRIPTION:	perform full frequency PCM synthesis | ||||
|  */ | ||||
| static | ||||
| void ICACHE_FLASH_ATTR  synth_full(struct mad_synth *synth, struct mad_frame const *frame, | ||||
| 		unsigned int nch, unsigned int ns) | ||||
| { | ||||
|   unsigned int phase, ch, s, sb, pe, po; | ||||
|   short int *pcm1, *pcm2; | ||||
|   mad_fixed_t (*filter)[2][2][16][8]; | ||||
|   mad_fixed_t (*sbsample)[36][32]; | ||||
|   register mad_fixed_t (*fe)[8], (*fx)[8], (*fo)[8]; | ||||
|   register mad_fixed_t const (*Dptr)[32], *ptr ; | ||||
|   register mad_fixed64hi_t hi; | ||||
|   register mad_fixed64lo_t lo; | ||||
|   mad_fixed_t raw_sample; | ||||
|   short int short_sample_buff[64]; //32];
 | ||||
| 
 | ||||
|   phase = synth->phase; | ||||
| 
 | ||||
|   for (s = 0; s < ns; ++s) | ||||
|   { | ||||
|     memset(short_sample_buff, 0x00, sizeof(short_sample_buff)); | ||||
| 
 | ||||
|     for (ch = 0; ch < nch; ++ch) | ||||
|     { | ||||
|     sbsample = (void*)&frame->sbsample[ch]; | ||||
|     filter   = &synth->filter[ch]; | ||||
|       pcm1     = short_sample_buff; | ||||
| 
 | ||||
|       dct32((*sbsample)[s], phase >> 1, | ||||
| 	    (*filter)[0][phase & 1], (*filter)[1][phase & 1]); | ||||
| 
 | ||||
|       pe = phase & ~1; | ||||
|       po = ((phase - 1) & 0xf) | 1; | ||||
| 
 | ||||
|       /* calculate 16 samples */ | ||||
| 
 | ||||
|       fe = &(*filter)[0][ phase & 1][0]; | ||||
|       fx = &(*filter)[0][~phase & 1][0]; | ||||
|       fo = &(*filter)[1][~phase & 1][0]; | ||||
| 
 | ||||
|       Dptr = &D[0]; | ||||
| 
 | ||||
|       ptr = *Dptr + po; | ||||
|       ML0(hi, lo, (*fx)[0], ptr[ 0]); | ||||
|       MLA(hi, lo, (*fx)[1], ptr[14]); | ||||
|       MLA(hi, lo, (*fx)[2], ptr[12]); | ||||
|       MLA(hi, lo, (*fx)[3], ptr[10]); | ||||
|       MLA(hi, lo, (*fx)[4], ptr[ 8]); | ||||
|       MLA(hi, lo, (*fx)[5], ptr[ 6]); | ||||
|       MLA(hi, lo, (*fx)[6], ptr[ 4]); | ||||
|       MLA(hi, lo, (*fx)[7], ptr[ 2]); | ||||
|       MLN(hi, lo); | ||||
| 
 | ||||
|       ptr = *Dptr + pe; | ||||
|       MLA(hi, lo, (*fe)[0], ptr[ 0]); | ||||
|       MLA(hi, lo, (*fe)[1], ptr[14]); | ||||
|       MLA(hi, lo, (*fe)[2], ptr[12]); | ||||
|       MLA(hi, lo, (*fe)[3], ptr[10]); | ||||
|       MLA(hi, lo, (*fe)[4], ptr[ 8]); | ||||
|       MLA(hi, lo, (*fe)[5], ptr[ 6]); | ||||
|       MLA(hi, lo, (*fe)[6], ptr[ 4]); | ||||
|       MLA(hi, lo, (*fe)[7], ptr[ 2]); | ||||
| 
 | ||||
|       raw_sample = SHIFT(MLZ(hi, lo)); | ||||
|       raw_sample = scale(raw_sample); | ||||
|       (*pcm1++) += (short int)raw_sample; | ||||
|       pcm2 = pcm1 + 30; | ||||
| 
 | ||||
|       for (sb = 1; sb < 16; ++sb) | ||||
|       { | ||||
| 	++fe; | ||||
| 	++Dptr; | ||||
| 
 | ||||
| 	/* D[32 - sb][i] == -D[sb][31 - i] */ | ||||
| 
 | ||||
| 	ptr = *Dptr + po; | ||||
| 	ML0(hi, lo, (*fo)[0], ptr[ 0]); | ||||
| 	MLA(hi, lo, (*fo)[1], ptr[14]); | ||||
| 	MLA(hi, lo, (*fo)[2], ptr[12]); | ||||
| 	MLA(hi, lo, (*fo)[3], ptr[10]); | ||||
| 	MLA(hi, lo, (*fo)[4], ptr[ 8]); | ||||
| 	MLA(hi, lo, (*fo)[5], ptr[ 6]); | ||||
| 	MLA(hi, lo, (*fo)[6], ptr[ 4]); | ||||
| 	MLA(hi, lo, (*fo)[7], ptr[ 2]); | ||||
| 	MLN(hi, lo); | ||||
| 
 | ||||
| 	ptr = *Dptr + pe; | ||||
| 	MLA(hi, lo, (*fe)[7], ptr[ 2]); | ||||
| 	MLA(hi, lo, (*fe)[6], ptr[ 4]); | ||||
| 	MLA(hi, lo, (*fe)[5], ptr[ 6]); | ||||
| 	MLA(hi, lo, (*fe)[4], ptr[ 8]); | ||||
| 	MLA(hi, lo, (*fe)[3], ptr[10]); | ||||
| 	MLA(hi, lo, (*fe)[2], ptr[12]); | ||||
| 	MLA(hi, lo, (*fe)[1], ptr[14]); | ||||
| 	MLA(hi, lo, (*fe)[0], ptr[ 0]); | ||||
| 
 | ||||
|         raw_sample = SHIFT(MLZ(hi, lo)); | ||||
|         raw_sample = scale(raw_sample); | ||||
|         (*pcm1++) += (short int)raw_sample; | ||||
| 
 | ||||
| 	ptr = *Dptr - pe; | ||||
| 	ML0(hi, lo, (*fe)[0], ptr[31 - 16]); | ||||
| 	MLA(hi, lo, (*fe)[1], ptr[31 - 14]); | ||||
| 	MLA(hi, lo, (*fe)[2], ptr[31 - 12]); | ||||
| 	MLA(hi, lo, (*fe)[3], ptr[31 - 10]); | ||||
| 	MLA(hi, lo, (*fe)[4], ptr[31 -  8]); | ||||
| 	MLA(hi, lo, (*fe)[5], ptr[31 -  6]); | ||||
| 	MLA(hi, lo, (*fe)[6], ptr[31 -  4]); | ||||
| 	MLA(hi, lo, (*fe)[7], ptr[31 -  2]); | ||||
| 
 | ||||
| 	ptr = *Dptr - po; | ||||
| 	MLA(hi, lo, (*fo)[7], ptr[31 -  2]); | ||||
| 	MLA(hi, lo, (*fo)[6], ptr[31 -  4]); | ||||
| 	MLA(hi, lo, (*fo)[5], ptr[31 -  6]); | ||||
| 	MLA(hi, lo, (*fo)[4], ptr[31 -  8]); | ||||
| 	MLA(hi, lo, (*fo)[3], ptr[31 - 10]); | ||||
| 	MLA(hi, lo, (*fo)[2], ptr[31 - 12]); | ||||
| 	MLA(hi, lo, (*fo)[1], ptr[31 - 14]); | ||||
| 	MLA(hi, lo, (*fo)[0], ptr[31 - 16]); | ||||
| 
 | ||||
|         raw_sample = SHIFT(MLZ(hi, lo)); | ||||
|         raw_sample = scale(raw_sample); | ||||
|         (*pcm2--) += (short int)raw_sample; | ||||
| 
 | ||||
| 	++fo; | ||||
|       } | ||||
| 
 | ||||
|       Dptr++; | ||||
| 
 | ||||
|       ptr = *Dptr + po; | ||||
|       ML0(hi, lo, (*fo)[0], ptr[ 0]); | ||||
|       MLA(hi, lo, (*fo)[1], ptr[14]); | ||||
|       MLA(hi, lo, (*fo)[2], ptr[12]); | ||||
|       MLA(hi, lo, (*fo)[3], ptr[10]); | ||||
|       MLA(hi, lo, (*fo)[4], ptr[ 8]); | ||||
|       MLA(hi, lo, (*fo)[5], ptr[ 6]); | ||||
|       MLA(hi, lo, (*fo)[6], ptr[ 4]); | ||||
|       MLA(hi, lo, (*fo)[7], ptr[ 2]); | ||||
| 
 | ||||
|       raw_sample = SHIFT(-MLZ(hi, lo)); | ||||
|       raw_sample = scale(raw_sample); | ||||
|       (*pcm1) += (short int)raw_sample; | ||||
| 
 | ||||
|     }  /* Channel For */ | ||||
| 
 | ||||
|     /* Render di un blocco */ | ||||
|     render_sample_block(short_sample_buff, sizeof(short_sample_buff)/sizeof(short int)); | ||||
| 
 | ||||
|       phase = (phase + 1) % 16; | ||||
| 
 | ||||
|   } /* Block for */ | ||||
| } | ||||
| #endif | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	synth->half() | ||||
|  * DESCRIPTION:	perform half frequency PCM synthesis | ||||
|  */ | ||||
| static | ||||
| void ICACHE_FLASH_ATTR synth_half(struct mad_synth *synth, struct mad_frame const *frame, | ||||
| 		unsigned int nch, unsigned int ns) | ||||
| { | ||||
|   unsigned int phase, ch, s, sb, pe, po; | ||||
|   short int *pcm1, *pcm2; | ||||
|   mad_fixed_t (*filter)[2][2][16][8]; | ||||
|   mad_fixed_t (*sbsample)[36][32]; | ||||
|   register mad_fixed_t (*fe)[8], (*fx)[8], (*fo)[8]; | ||||
|   register mad_fixed_t const (*Dptr)[32], *ptr ; | ||||
|   register mad_fixed64hi_t hi; | ||||
|   register mad_fixed64lo_t lo; | ||||
|   mad_fixed_t raw_sample; | ||||
|   short int short_sample_buff[16]; | ||||
| 
 | ||||
|   phase = synth->phase; | ||||
| 
 | ||||
|   for (s = 0; s < ns; ++s) | ||||
|   { | ||||
|     memset (short_sample_buff, 0x00, sizeof(short_sample_buff)); | ||||
| 
 | ||||
|     for (ch = 0; ch < nch; ++ch) | ||||
|     { | ||||
|     sbsample = (void *)&frame->sbsample[ch]; | ||||
|     filter   = &synth->filter[ch]; | ||||
|       pcm1     = short_sample_buff; | ||||
| 
 | ||||
|       dct32((*sbsample)[s], phase >> 1, | ||||
| 	    (*filter)[0][phase & 1], (*filter)[1][phase & 1]); | ||||
| 
 | ||||
|       pe = phase & ~1; | ||||
|       po = ((phase - 1) & 0xf) | 1; | ||||
| 
 | ||||
|       /* calculate 16 samples */ | ||||
| 
 | ||||
|       fe = &(*filter)[0][ phase & 1][0]; | ||||
|       fx = &(*filter)[0][~phase & 1][0]; | ||||
|       fo = &(*filter)[1][~phase & 1][0]; | ||||
| 
 | ||||
|       Dptr = &D[0]; | ||||
| 
 | ||||
|       ptr = *Dptr + po; | ||||
|       ML0(hi, lo, (*fx)[0], ptr[ 0]); | ||||
|       MLA(hi, lo, (*fx)[1], ptr[14]); | ||||
|       MLA(hi, lo, (*fx)[2], ptr[12]); | ||||
|       MLA(hi, lo, (*fx)[3], ptr[10]); | ||||
|       MLA(hi, lo, (*fx)[4], ptr[ 8]); | ||||
|       MLA(hi, lo, (*fx)[5], ptr[ 6]); | ||||
|       MLA(hi, lo, (*fx)[6], ptr[ 4]); | ||||
|       MLA(hi, lo, (*fx)[7], ptr[ 2]); | ||||
|       MLN(hi, lo); | ||||
| 
 | ||||
|       ptr = *Dptr + pe; | ||||
|       MLA(hi, lo, (*fe)[0], ptr[ 0]); | ||||
|       MLA(hi, lo, (*fe)[1], ptr[14]); | ||||
|       MLA(hi, lo, (*fe)[2], ptr[12]); | ||||
|       MLA(hi, lo, (*fe)[3], ptr[10]); | ||||
|       MLA(hi, lo, (*fe)[4], ptr[ 8]); | ||||
|       MLA(hi, lo, (*fe)[5], ptr[ 6]); | ||||
|       MLA(hi, lo, (*fe)[6], ptr[ 4]); | ||||
|       MLA(hi, lo, (*fe)[7], ptr[ 2]); | ||||
| 
 | ||||
|       raw_sample = SHIFT(MLZ(hi, lo)); | ||||
|       raw_sample = scale(raw_sample); | ||||
|       (*pcm1++) += (short int)raw_sample; | ||||
|       pcm2 = pcm1 + 14; | ||||
| 
 | ||||
|       for (sb = 1; sb < 16; ++sb) | ||||
|       { | ||||
| 	++fe; | ||||
| 	++Dptr; | ||||
| 
 | ||||
| 	/* D[32 - sb][i] == -D[sb][31 - i] */ | ||||
| 
 | ||||
| 	  ptr = *Dptr + po; | ||||
| 	  ML0(hi, lo, (*fo)[0], ptr[ 0]); | ||||
| 	  MLA(hi, lo, (*fo)[1], ptr[14]); | ||||
| 	  MLA(hi, lo, (*fo)[2], ptr[12]); | ||||
| 	  MLA(hi, lo, (*fo)[3], ptr[10]); | ||||
| 	  MLA(hi, lo, (*fo)[4], ptr[ 8]); | ||||
| 	  MLA(hi, lo, (*fo)[5], ptr[ 6]); | ||||
| 	  MLA(hi, lo, (*fo)[6], ptr[ 4]); | ||||
| 	  MLA(hi, lo, (*fo)[7], ptr[ 2]); | ||||
| 	  MLN(hi, lo); | ||||
| 
 | ||||
| 	  ptr = *Dptr + pe; | ||||
| 	  MLA(hi, lo, (*fe)[7], ptr[ 2]); | ||||
| 	  MLA(hi, lo, (*fe)[6], ptr[ 4]); | ||||
| 	  MLA(hi, lo, (*fe)[5], ptr[ 6]); | ||||
| 	  MLA(hi, lo, (*fe)[4], ptr[ 8]); | ||||
| 	  MLA(hi, lo, (*fe)[3], ptr[10]); | ||||
| 	  MLA(hi, lo, (*fe)[2], ptr[12]); | ||||
| 	  MLA(hi, lo, (*fe)[1], ptr[14]); | ||||
| 	  MLA(hi, lo, (*fe)[0], ptr[ 0]); | ||||
| 
 | ||||
|         raw_sample = SHIFT(MLZ(hi, lo)); | ||||
|         raw_sample = scale(raw_sample); | ||||
|         (*pcm1++) += (short int)raw_sample; | ||||
| 
 | ||||
| 	  ptr = *Dptr - pe; | ||||
|         ML0(hi, lo, (*fe)[0], ptr[31 - 16]); | ||||
| 	  MLA(hi, lo, (*fe)[1], ptr[31 - 14]); | ||||
| 	  MLA(hi, lo, (*fe)[2], ptr[31 - 12]); | ||||
| 	  MLA(hi, lo, (*fe)[3], ptr[31 - 10]); | ||||
| 	  MLA(hi, lo, (*fe)[4], ptr[31 -  8]); | ||||
| 	  MLA(hi, lo, (*fe)[5], ptr[31 -  6]); | ||||
| 	  MLA(hi, lo, (*fe)[6], ptr[31 -  4]); | ||||
| 	  MLA(hi, lo, (*fe)[7], ptr[31 -  2]); | ||||
| 
 | ||||
|         ptr = *Dptr - po; | ||||
|         MLA(hi, lo, (*fo)[7], ptr[31 -  2]); | ||||
|         MLA(hi, lo, (*fo)[6], ptr[31 -  4]); | ||||
|         MLA(hi, lo, (*fo)[5], ptr[31 -  6]); | ||||
|         MLA(hi, lo, (*fo)[4], ptr[31 -  8]); | ||||
|         MLA(hi, lo, (*fo)[3], ptr[31 - 10]); | ||||
|         MLA(hi, lo, (*fo)[2], ptr[31 - 12]); | ||||
|         MLA(hi, lo, (*fo)[1], ptr[31 - 14]); | ||||
|         MLA(hi, lo, (*fo)[0], ptr[31 - 16]); | ||||
| 
 | ||||
|         raw_sample = SHIFT(MLZ(hi, lo)); | ||||
|         raw_sample = scale(raw_sample); | ||||
|         (*pcm2--) += (short int)raw_sample; | ||||
| 
 | ||||
| 	++fo; | ||||
|       } | ||||
| 
 | ||||
|       Dptr++; | ||||
| 
 | ||||
|       ptr = *Dptr + po; | ||||
|       ML0(hi, lo, (*fo)[0], ptr[ 0]); | ||||
|       MLA(hi, lo, (*fo)[1], ptr[14]); | ||||
|       MLA(hi, lo, (*fo)[2], ptr[12]); | ||||
|       MLA(hi, lo, (*fo)[3], ptr[10]); | ||||
|       MLA(hi, lo, (*fo)[4], ptr[ 8]); | ||||
|       MLA(hi, lo, (*fo)[5], ptr[ 6]); | ||||
|       MLA(hi, lo, (*fo)[6], ptr[ 4]); | ||||
|       MLA(hi, lo, (*fo)[7], ptr[ 2]); | ||||
| 
 | ||||
|       raw_sample = SHIFT(-MLZ(hi, lo)); | ||||
|       raw_sample = scale(raw_sample); | ||||
|       (*pcm1) += (short int)raw_sample; | ||||
| 
 | ||||
|     } /* Channel For */ | ||||
| 
 | ||||
|     /* Block render */ | ||||
|     render_sample_block(short_sample_buff, sizeof(short_sample_buff)/sizeof(short int)); | ||||
| 
 | ||||
|       phase = (phase + 1) % 16; | ||||
| 
 | ||||
|   }/* Block For */ | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	synth->frame() | ||||
|  * DESCRIPTION:	perform PCM synthesis of frame subband samples | ||||
|  */ | ||||
| void ICACHE_FLASH_ATTR mad_synth_frame(struct mad_synth *synth, struct mad_frame const *frame) | ||||
| { | ||||
|   unsigned int nch, ns; | ||||
|   void (*synth_frame)(struct mad_synth *, struct mad_frame const *, | ||||
| 		      unsigned int, unsigned int); | ||||
| 
 | ||||
|   nch = MAD_NCHANNELS(&frame->header); | ||||
|   ns  = MAD_NSBSAMPLES(&frame->header); | ||||
| 
 | ||||
|   synth->pcm.samplerate = frame->header.samplerate; | ||||
|   synth->pcm.channels   = nch; | ||||
| //  synth->pcm.length     = 32 * ns;
 | ||||
|   synth->pcm.length     = 128 * ns; | ||||
| 
 | ||||
|   synth_frame = synth_full; | ||||
| 
 | ||||
|   if (frame->options & MAD_OPTION_HALFSAMPLERATE) { | ||||
|     synth->pcm.samplerate /= 2; | ||||
|     synth->pcm.length     /= 2; | ||||
|     synth_frame = synth_half; | ||||
|   } | ||||
| 
 | ||||
|   set_dac_sample_rate(synth->pcm.samplerate); | ||||
| 
 | ||||
|   synth_frame(synth, frame, nch, ns); | ||||
| 
 | ||||
|   synth->phase = (synth->phase + ns) % 16; | ||||
| } | ||||
							
								
								
									
										485
									
								
								project/src/mad/timer.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										485
									
								
								project/src/mad/timer.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,485 @@ | |||
| /*
 | ||||
|  * libmad - MPEG audio decoder library | ||||
|  * Copyright (C) 2000-2004 Underbit Technologies, Inc. | ||||
|  * | ||||
|  * This program is free software; you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|  * the Free Software Foundation; either version 2 of the License, or | ||||
|  * (at your option) any later version. | ||||
|  * | ||||
|  * This program 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.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  * | ||||
|  * You should have received a copy of the GNU General Public License | ||||
|  * along with this program; if not, write to the Free Software | ||||
|  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA | ||||
|  * | ||||
|  * $Id: timer.c,v 1.18 2004/01/23 09:41:33 rob Exp $ | ||||
|  */ | ||||
| 
 | ||||
| # ifdef HAVE_CONFIG_H | ||||
| #  include "config.h" | ||||
| # endif | ||||
| 
 | ||||
| # include "global.h" | ||||
| 
 | ||||
| # include <stdio.h> | ||||
| 
 | ||||
| # ifdef HAVE_ASSERT_H | ||||
| #  include <assert.h> | ||||
| # endif | ||||
| 
 | ||||
| # include "timer.h" | ||||
| 
 | ||||
| mad_timer_t const mad_timer_zero = { 0, 0 }; | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	timer->compare() | ||||
|  * DESCRIPTION:	indicate relative order of two timers | ||||
|  */ | ||||
| int ICACHE_FLASH_ATTR mad_timer_compare(mad_timer_t timer1, mad_timer_t timer2) | ||||
| { | ||||
|   signed long diff; | ||||
| 
 | ||||
|   diff = timer1.seconds - timer2.seconds; | ||||
|   if (diff < 0) | ||||
|     return -1; | ||||
|   else if (diff > 0) | ||||
|     return +1; | ||||
| 
 | ||||
|   diff = timer1.fraction - timer2.fraction; | ||||
|   if (diff < 0) | ||||
|     return -1; | ||||
|   else if (diff > 0) | ||||
|     return +1; | ||||
| 
 | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	timer->negate() | ||||
|  * DESCRIPTION:	invert the sign of a timer | ||||
|  */ | ||||
| void ICACHE_FLASH_ATTR mad_timer_negate(mad_timer_t *timer) | ||||
| { | ||||
|   timer->seconds = -timer->seconds; | ||||
| 
 | ||||
|   if (timer->fraction) { | ||||
|     timer->seconds -= 1; | ||||
|     timer->fraction = MAD_TIMER_RESOLUTION - timer->fraction; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	timer->abs() | ||||
|  * DESCRIPTION:	return the absolute value of a timer | ||||
|  */ | ||||
| mad_timer_t mad_timer_abs(mad_timer_t timer) | ||||
| { | ||||
|   if (timer.seconds < 0) | ||||
|     mad_timer_negate(&timer); | ||||
| 
 | ||||
|   return timer; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	reduce_timer() | ||||
|  * DESCRIPTION:	carry timer fraction into seconds | ||||
|  */ | ||||
| static | ||||
| void ICACHE_FLASH_ATTR reduce_timer(mad_timer_t *timer) | ||||
| { | ||||
|   timer->seconds  += timer->fraction / MAD_TIMER_RESOLUTION; | ||||
|   timer->fraction %= MAD_TIMER_RESOLUTION; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	gcd() | ||||
|  * DESCRIPTION:	compute greatest common denominator | ||||
|  */ | ||||
| static | ||||
| unsigned long ICACHE_FLASH_ATTR gcd(unsigned long num1, unsigned long num2) | ||||
| { | ||||
|   unsigned long tmp; | ||||
| 
 | ||||
|   while (num2) { | ||||
|     tmp  = num2; | ||||
|     num2 = num1 % num2; | ||||
|     num1 = tmp; | ||||
|   } | ||||
| 
 | ||||
|   return num1; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	reduce_rational() | ||||
|  * DESCRIPTION:	convert rational expression to lowest terms | ||||
|  */ | ||||
| static | ||||
| void ICACHE_FLASH_ATTR reduce_rational(unsigned long *numer, unsigned long *denom) | ||||
| { | ||||
|   unsigned long factor; | ||||
| 
 | ||||
|   factor = gcd(*numer, *denom); | ||||
| 
 | ||||
|   //assert(factor != 0);
 | ||||
| 
 | ||||
|   *numer /= factor; | ||||
|   *denom /= factor; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	scale_rational() | ||||
|  * DESCRIPTION:	solve numer/denom == ?/scale avoiding overflowing | ||||
|  */ | ||||
| static | ||||
| unsigned long ICACHE_FLASH_ATTR scale_rational(unsigned long numer, unsigned long denom, | ||||
| 			     unsigned long scale) | ||||
| { | ||||
|   reduce_rational(&numer, &denom); | ||||
|   reduce_rational(&scale, &denom); | ||||
| 
 | ||||
|   //assert(denom != 0);
 | ||||
| 
 | ||||
|   if (denom < scale) | ||||
|     return numer * (scale / denom) + numer * (scale % denom) / denom; | ||||
|   if (denom < numer) | ||||
|     return scale * (numer / denom) + scale * (numer % denom) / denom; | ||||
| 
 | ||||
|   return numer * scale / denom; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	timer->set() | ||||
|  * DESCRIPTION:	set timer to specific (positive) value | ||||
|  */ | ||||
| void ICACHE_FLASH_ATTR mad_timer_set(mad_timer_t *timer, unsigned long seconds, | ||||
| 		   unsigned long numer, unsigned long denom) | ||||
| { | ||||
|   timer->seconds = seconds; | ||||
|   if (numer >= denom && denom > 0) { | ||||
|     timer->seconds += numer / denom; | ||||
|     numer %= denom; | ||||
|   } | ||||
| 
 | ||||
|   switch (denom) { | ||||
|   case 0: | ||||
|   case 1: | ||||
|     timer->fraction = 0; | ||||
|     break; | ||||
| 
 | ||||
|   case MAD_TIMER_RESOLUTION: | ||||
|     timer->fraction = numer; | ||||
|     break; | ||||
| 
 | ||||
|   case 1000: | ||||
|     timer->fraction = numer * (MAD_TIMER_RESOLUTION /  1000); | ||||
|     break; | ||||
| 
 | ||||
|   case 8000: | ||||
|     timer->fraction = numer * (MAD_TIMER_RESOLUTION /  8000); | ||||
|     break; | ||||
| 
 | ||||
|   case 11025: | ||||
|     timer->fraction = numer * (MAD_TIMER_RESOLUTION / 11025); | ||||
|     break; | ||||
| 
 | ||||
|   case 12000: | ||||
|     timer->fraction = numer * (MAD_TIMER_RESOLUTION / 12000); | ||||
|     break; | ||||
| 
 | ||||
|   case 16000: | ||||
|     timer->fraction = numer * (MAD_TIMER_RESOLUTION / 16000); | ||||
|     break; | ||||
| 
 | ||||
|   case 22050: | ||||
|     timer->fraction = numer * (MAD_TIMER_RESOLUTION / 22050); | ||||
|     break; | ||||
| 
 | ||||
|   case 24000: | ||||
|     timer->fraction = numer * (MAD_TIMER_RESOLUTION / 24000); | ||||
|     break; | ||||
| 
 | ||||
|   case 32000: | ||||
|     timer->fraction = numer * (MAD_TIMER_RESOLUTION / 32000); | ||||
|     break; | ||||
| 
 | ||||
|   case 44100: | ||||
|     timer->fraction = numer * (MAD_TIMER_RESOLUTION / 44100); | ||||
|     break; | ||||
| 
 | ||||
|   case 48000: | ||||
|     timer->fraction = numer * (MAD_TIMER_RESOLUTION / 48000); | ||||
|     break; | ||||
| 
 | ||||
|   default: | ||||
|     timer->fraction = scale_rational(numer, denom, MAD_TIMER_RESOLUTION); | ||||
|     break; | ||||
|   } | ||||
| 
 | ||||
|   if (timer->fraction >= MAD_TIMER_RESOLUTION) | ||||
|     reduce_timer(timer); | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	timer->add() | ||||
|  * DESCRIPTION:	add one timer to another | ||||
|  */ | ||||
| void ICACHE_FLASH_ATTR mad_timer_add(mad_timer_t *timer, mad_timer_t incr) | ||||
| { | ||||
|   timer->seconds  += incr.seconds; | ||||
|   timer->fraction += incr.fraction; | ||||
| 
 | ||||
|   if (timer->fraction >= MAD_TIMER_RESOLUTION) | ||||
|     reduce_timer(timer); | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	timer->multiply() | ||||
|  * DESCRIPTION:	multiply a timer by a scalar value | ||||
|  */ | ||||
| void ICACHE_FLASH_ATTR mad_timer_multiply(mad_timer_t *timer, signed long scalar) | ||||
| { | ||||
|   mad_timer_t addend; | ||||
|   unsigned long factor; | ||||
| 
 | ||||
|   factor = scalar; | ||||
|   if (scalar < 0) { | ||||
|     factor = -scalar; | ||||
|     mad_timer_negate(timer); | ||||
|   } | ||||
| 
 | ||||
|   addend = *timer; | ||||
|   *timer = mad_timer_zero; | ||||
| 
 | ||||
|   while (factor) { | ||||
|     if (factor & 1) | ||||
|       mad_timer_add(timer, addend); | ||||
| 
 | ||||
|     mad_timer_add(&addend, addend); | ||||
|     factor >>= 1; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	timer->count() | ||||
|  * DESCRIPTION:	return timer value in selected units | ||||
|  */ | ||||
| signed long ICACHE_FLASH_ATTR mad_timer_count(mad_timer_t timer, enum mad_units units) | ||||
| { | ||||
|   switch (units) { | ||||
|   case MAD_UNITS_HOURS: | ||||
|     return timer.seconds / 60 / 60; | ||||
| 
 | ||||
|   case MAD_UNITS_MINUTES: | ||||
|     return timer.seconds / 60; | ||||
| 
 | ||||
|   case MAD_UNITS_SECONDS: | ||||
|     return timer.seconds; | ||||
| 
 | ||||
|   case MAD_UNITS_DECISECONDS: | ||||
|   case MAD_UNITS_CENTISECONDS: | ||||
|   case MAD_UNITS_MILLISECONDS: | ||||
| 
 | ||||
|   case MAD_UNITS_8000_HZ: | ||||
|   case MAD_UNITS_11025_HZ: | ||||
|   case MAD_UNITS_12000_HZ: | ||||
|   case MAD_UNITS_16000_HZ: | ||||
|   case MAD_UNITS_22050_HZ: | ||||
|   case MAD_UNITS_24000_HZ: | ||||
|   case MAD_UNITS_32000_HZ: | ||||
|   case MAD_UNITS_44100_HZ: | ||||
|   case MAD_UNITS_48000_HZ: | ||||
| 
 | ||||
|   case MAD_UNITS_24_FPS: | ||||
|   case MAD_UNITS_25_FPS: | ||||
|   case MAD_UNITS_30_FPS: | ||||
|   case MAD_UNITS_48_FPS: | ||||
|   case MAD_UNITS_50_FPS: | ||||
|   case MAD_UNITS_60_FPS: | ||||
|   case MAD_UNITS_75_FPS: | ||||
|     return timer.seconds * (signed long) units + | ||||
|       (signed long) scale_rational(timer.fraction, MAD_TIMER_RESOLUTION, | ||||
| 				   units); | ||||
| 
 | ||||
|   case MAD_UNITS_23_976_FPS: | ||||
|   case MAD_UNITS_24_975_FPS: | ||||
|   case MAD_UNITS_29_97_FPS: | ||||
|   case MAD_UNITS_47_952_FPS: | ||||
|   case MAD_UNITS_49_95_FPS: | ||||
|   case MAD_UNITS_59_94_FPS: | ||||
|     return (mad_timer_count(timer, -units) + 1) * 1000 / 1001; | ||||
|   } | ||||
| 
 | ||||
|   /* unsupported units */ | ||||
|   return 0; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	timer->fraction() | ||||
|  * DESCRIPTION:	return fractional part of timer in arbitrary terms | ||||
|  */ | ||||
| unsigned long ICACHE_FLASH_ATTR mad_timer_fraction(mad_timer_t timer, unsigned long denom) | ||||
| { | ||||
|   timer = mad_timer_abs(timer); | ||||
| 
 | ||||
|   switch (denom) { | ||||
|   case 0: | ||||
|     return timer.fraction ? | ||||
|       MAD_TIMER_RESOLUTION / timer.fraction : MAD_TIMER_RESOLUTION + 1; | ||||
| 
 | ||||
|   case MAD_TIMER_RESOLUTION: | ||||
|     return timer.fraction; | ||||
| 
 | ||||
|   default: | ||||
|     return scale_rational(timer.fraction, MAD_TIMER_RESOLUTION, denom); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * NAME:	timer->string() | ||||
|  * DESCRIPTION:	write a string representation of a timer using a template | ||||
|  */ | ||||
| void ICACHE_FLASH_ATTR mad_timer_string(mad_timer_t timer, | ||||
| 		      char *dest, char const *format, enum mad_units units, | ||||
| 		      enum mad_units fracunits, unsigned long subparts) | ||||
| { | ||||
|   unsigned long hours, minutes, seconds, sub; | ||||
|   unsigned int frac; | ||||
| 
 | ||||
|   timer = mad_timer_abs(timer); | ||||
| 
 | ||||
|   seconds = timer.seconds; | ||||
|   frac = sub = 0; | ||||
| 
 | ||||
|   switch (fracunits) { | ||||
|   case MAD_UNITS_HOURS: | ||||
|   case MAD_UNITS_MINUTES: | ||||
|   case MAD_UNITS_SECONDS: | ||||
|     break; | ||||
| 
 | ||||
|   case MAD_UNITS_DECISECONDS: | ||||
|   case MAD_UNITS_CENTISECONDS: | ||||
|   case MAD_UNITS_MILLISECONDS: | ||||
| 
 | ||||
|   case MAD_UNITS_8000_HZ: | ||||
|   case MAD_UNITS_11025_HZ: | ||||
|   case MAD_UNITS_12000_HZ: | ||||
|   case MAD_UNITS_16000_HZ: | ||||
|   case MAD_UNITS_22050_HZ: | ||||
|   case MAD_UNITS_24000_HZ: | ||||
|   case MAD_UNITS_32000_HZ: | ||||
|   case MAD_UNITS_44100_HZ: | ||||
|   case MAD_UNITS_48000_HZ: | ||||
| 
 | ||||
|   case MAD_UNITS_24_FPS: | ||||
|   case MAD_UNITS_25_FPS: | ||||
|   case MAD_UNITS_30_FPS: | ||||
|   case MAD_UNITS_48_FPS: | ||||
|   case MAD_UNITS_50_FPS: | ||||
|   case MAD_UNITS_60_FPS: | ||||
|   case MAD_UNITS_75_FPS: | ||||
|     { | ||||
|       unsigned long denom; | ||||
| 
 | ||||
|       denom = MAD_TIMER_RESOLUTION / fracunits; | ||||
| 
 | ||||
|       frac = timer.fraction / denom; | ||||
|       sub  = scale_rational(timer.fraction % denom, denom, subparts); | ||||
|     } | ||||
|     break; | ||||
| 
 | ||||
|   case MAD_UNITS_23_976_FPS: | ||||
|   case MAD_UNITS_24_975_FPS: | ||||
|   case MAD_UNITS_29_97_FPS: | ||||
|   case MAD_UNITS_47_952_FPS: | ||||
|   case MAD_UNITS_49_95_FPS: | ||||
|   case MAD_UNITS_59_94_FPS: | ||||
|     /* drop-frame encoding */ | ||||
|     /* N.B. this is only well-defined for MAD_UNITS_29_97_FPS */ | ||||
|     { | ||||
|       unsigned long frame, cycle, d, m; | ||||
| 
 | ||||
|       frame = mad_timer_count(timer, fracunits); | ||||
| 
 | ||||
|       cycle = -fracunits * 60 * 10 - (10 - 1) * 2; | ||||
| 
 | ||||
|       d = frame / cycle; | ||||
|       m = frame % cycle; | ||||
|       frame += (10 - 1) * 2 * d; | ||||
|       if (m > 2) | ||||
| 	frame += 2 * ((m - 2) / (cycle / 10)); | ||||
| 
 | ||||
|       frac    = frame % -fracunits; | ||||
|       seconds = frame / -fracunits; | ||||
|     } | ||||
|     break; | ||||
|   } | ||||
| 
 | ||||
|   switch (units) { | ||||
|   case MAD_UNITS_HOURS: | ||||
|     minutes = seconds / 60; | ||||
|     hours   = minutes / 60; | ||||
| 
 | ||||
|     sprintf(dest, format, | ||||
| 	    hours, | ||||
| 	    (unsigned int) (minutes % 60), | ||||
| 	    (unsigned int) (seconds % 60), | ||||
| 	    frac, sub); | ||||
|     break; | ||||
| 
 | ||||
|   case MAD_UNITS_MINUTES: | ||||
|     minutes = seconds / 60; | ||||
| 
 | ||||
|     sprintf(dest, format, | ||||
| 	    minutes, | ||||
| 	    (unsigned int) (seconds % 60), | ||||
| 	    frac, sub); | ||||
|     break; | ||||
| 
 | ||||
|   case MAD_UNITS_SECONDS: | ||||
|     sprintf(dest, format, | ||||
| 	    seconds, | ||||
| 	    frac, sub); | ||||
|     break; | ||||
| 
 | ||||
|   case MAD_UNITS_23_976_FPS: | ||||
|   case MAD_UNITS_24_975_FPS: | ||||
|   case MAD_UNITS_29_97_FPS: | ||||
|   case MAD_UNITS_47_952_FPS: | ||||
|   case MAD_UNITS_49_95_FPS: | ||||
|   case MAD_UNITS_59_94_FPS: | ||||
|     if (fracunits < 0) { | ||||
|       /* not yet implemented */ | ||||
|       sub = 0; | ||||
|     } | ||||
| 
 | ||||
|     /* fall through */ | ||||
| 
 | ||||
|   case MAD_UNITS_DECISECONDS: | ||||
|   case MAD_UNITS_CENTISECONDS: | ||||
|   case MAD_UNITS_MILLISECONDS: | ||||
| 
 | ||||
|   case MAD_UNITS_8000_HZ: | ||||
|   case MAD_UNITS_11025_HZ: | ||||
|   case MAD_UNITS_12000_HZ: | ||||
|   case MAD_UNITS_16000_HZ: | ||||
|   case MAD_UNITS_22050_HZ: | ||||
|   case MAD_UNITS_24000_HZ: | ||||
|   case MAD_UNITS_32000_HZ: | ||||
|   case MAD_UNITS_44100_HZ: | ||||
|   case MAD_UNITS_48000_HZ: | ||||
| 
 | ||||
|   case MAD_UNITS_24_FPS: | ||||
|   case MAD_UNITS_25_FPS: | ||||
|   case MAD_UNITS_30_FPS: | ||||
|   case MAD_UNITS_48_FPS: | ||||
|   case MAD_UNITS_50_FPS: | ||||
|   case MAD_UNITS_60_FPS: | ||||
|   case MAD_UNITS_75_FPS: | ||||
|     sprintf(dest, format, mad_timer_count(timer, units), sub); | ||||
|     break; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										386
									
								
								project/src/user/atcmd_user.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										386
									
								
								project/src/user/atcmd_user.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,386 @@ | |||
| #include <platform_opts.h> | ||||
| 
 | ||||
| #ifdef CONFIG_AT_USR | ||||
| 
 | ||||
| #include "FreeRTOS.h" | ||||
| #include "task.h" | ||||
| #include "semphr.h" | ||||
| #include "at_cmd/log_service.h" | ||||
| #include "at_cmd/atcmd_wifi.h" | ||||
| #include <lwip_netconf.h> | ||||
| #include "tcpip.h" | ||||
| #include <dhcp/dhcps.h> | ||||
| #include <wifi/wifi_conf.h> | ||||
| #include <wifi/wifi_util.h> | ||||
| #include "tcm_heap.h" | ||||
| #include "user/atcmd_user.h" | ||||
| #include "user/playerconfig.h" | ||||
| 
 | ||||
| rtw_mode_t wifi_mode = RTW_MODE_STA; | ||||
| mp3_server_setings mp3_serv = {0,{0}}; //{ PLAY_PORT, { PLAY_SERVER }};
 | ||||
| 
 | ||||
| #define DEBUG_AT_USER_LEVEL 1 | ||||
| 
 | ||||
| /******************************************************************************/ | ||||
| /*
 | ||||
| #define	_AT_WLAN_SET_SSID_          "ATW0" | ||||
| #define	_AT_WLAN_SET_PASSPHRASE_    "ATW1" | ||||
| #define	_AT_WLAN_SET_KEY_ID_        "ATW2" | ||||
| #define	_AT_WLAN_JOIN_NET_          "ATWC" | ||||
| #define	_AT_WLAN_SET_MP3_URL_       "ATWS" | ||||
| */ | ||||
| //extern struct netif xnetif[NET_IF_NUM];
 | ||||
| 
 | ||||
| /* fastconnect use wifi AT command. Not init_wifi_struct when log service disabled
 | ||||
|  * static initialize all values for using fastconnect when log service disabled | ||||
|  */ | ||||
| static rtw_network_info_t wifi = { | ||||
| 	{0},    // ssid
 | ||||
| 	{0},    // bssid
 | ||||
| 	0,      // security
 | ||||
| 	NULL,   // password
 | ||||
| 	0,      // password len
 | ||||
| 	-1      // key id
 | ||||
| }; | ||||
| 
 | ||||
| static rtw_ap_info_t ap = {0}; | ||||
| static unsigned char password[65] = {0}; | ||||
| 
 | ||||
| _WEAK void connect_start(void) | ||||
| { | ||||
| } | ||||
| 
 | ||||
| _WEAK void connect_close(void) | ||||
| { | ||||
| } | ||||
| 
 | ||||
| static void init_wifi_struct(void) | ||||
| { | ||||
| 	memset(wifi.ssid.val, 0, sizeof(wifi.ssid.val)); | ||||
| 	memset(wifi.bssid.octet, 0, ETH_ALEN);	 | ||||
| 	memset(password, 0, sizeof(password)); | ||||
| 	wifi.ssid.len = 0; | ||||
| 	wifi.password = NULL; | ||||
| 	wifi.password_len = 0; | ||||
| 	wifi.key_id = -1; | ||||
| 	memset(ap.ssid.val, 0, sizeof(ap.ssid.val)); | ||||
| 	ap.ssid.len = 0; | ||||
| 	ap.password = NULL; | ||||
| 	ap.password_len = 0; | ||||
| 	ap.channel = 1; | ||||
| } | ||||
| 
 | ||||
| void fATW0(void *arg){ | ||||
| 	if(!arg){ | ||||
| 		printf("ATW0: Usage: ATW0=SSID\n"); | ||||
| 		goto exit; | ||||
| 	} | ||||
| #if	DEBUG_AT_USER_LEVEL > 1 | ||||
| 	printf("ATW0: %s\n", (char*)arg); | ||||
| #endif | ||||
| 	strcpy((char *)wifi.ssid.val, (char*)arg); | ||||
| 	wifi.ssid.len = strlen((char*)arg); | ||||
| exit: | ||||
| 	return; | ||||
| } | ||||
| 
 | ||||
| void fATW1(void *arg){ | ||||
| #if	DEBUG_AT_USER_LEVEL > 1 | ||||
|     printf("ATW1: %s\n", (char*)arg); | ||||
| #endif | ||||
| 	strcpy((char *)password, (char*)arg); | ||||
| 	wifi.password = password; | ||||
| 	wifi.password_len = strlen((char*)arg); | ||||
| 	return;	 | ||||
| } | ||||
| 
 | ||||
| void fATW2(void *arg){ | ||||
| #if	DEBUG_AT_USER_LEVEL > 1 | ||||
| 	printf("ATW2: %s\n", (char*)arg); | ||||
| #endif | ||||
| 	if((strlen((const char *)arg) != 1 ) || (*(char*)arg <'0' ||*(char*)arg >'3')) { | ||||
| 		printf("ATW2: Wrong WEP key id. Must be one of 0,1,2, or 3.\n"); | ||||
| 		return; | ||||
| 	} | ||||
| 	wifi.key_id = atoi((const char *)(arg)); | ||||
| 	return; | ||||
| } | ||||
| 
 | ||||
| // Test
 | ||||
| void fATST(void *arg){ | ||||
| 	extern u8 __HeapLimit, __StackTop; | ||||
| 	extern struct Heap g_tcm_heap; | ||||
| 		//DBG_INFO_MSG_ON(_DBG_TCM_HEAP_); // On Debug TCM MEM
 | ||||
| #if	DEBUG_AT_USER_LEVEL > 1 | ||||
| 		printf("ATST: Mem info:\n"); | ||||
| #endif | ||||
| //		vPortFree(pvPortMalloc(4)); // Init RAM heap
 | ||||
| 		printf("\nCLK CPU\t\t%d Hz\nRAM heap\t%d bytes\nRAM free\t%d bytes\nTCM heap\t%d bytes\n", | ||||
| 				HalGetCpuClk(), xPortGetFreeHeapSize(), (int)&__StackTop - (int)&__HeapLimit, tcm_heap_freeSpace()); | ||||
| 		printf("TCM ps_monitor\t%d bytes\n", 0x20000000 - (u32)&tcm_heap - tcm_heap_size); | ||||
| 		dump_mem_block_list(); | ||||
| 		u32 saved = ConfigDebugInfo; | ||||
| 		DBG_INFO_MSG_ON(_DBG_TCM_HEAP_); // On Debug TCM MEM
 | ||||
| 		tcm_heap_dump(); | ||||
| 		ConfigDebugInfo = saved; | ||||
| 		printf("\n"); | ||||
| #if (configGENERATE_RUN_TIME_STATS == 1) | ||||
| 		char *cBuffer = pvPortMalloc(512); | ||||
| 		if(cBuffer != NULL) { | ||||
| 			vTaskGetRunTimeStats((char *)cBuffer); | ||||
| 			printf("%s", cBuffer); | ||||
| 		} | ||||
| 		vPortFree(cBuffer); | ||||
| #endif | ||||
| } | ||||
| 
 | ||||
| int mp3_cfg_read(void) | ||||
| { | ||||
| 	bzero(&mp3_serv, sizeof(mp3_serv)); | ||||
| 	if(flash_read_cfg(mp3_serv, 0x5000, sizeof(mp3_serv.port) + 2) >= sizeof(mp3_serv.port) + 2) { | ||||
| 		mp3_serv.port = PLAY_PORT; | ||||
| 		strcpy(mp3_serv.url, PLAY_SERVER); | ||||
| 	} | ||||
| 	return mp3_serv.port; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| // MP3 Set server, Close connect
 | ||||
| void fATWS(void *arg){ | ||||
| 	int   argc           = 0; | ||||
| 	char *argv[MAX_ARGC] = {0}; | ||||
| 	if(arg) { | ||||
|        	argc = parse_param(arg, argv); | ||||
|     	if (argc == 2) { | ||||
|     		if(argv[1][0] == '?') { | ||||
| 			    printf("ATWS: %s,%d\n", mp3_serv.url, mp3_serv.port); | ||||
|     		    return; | ||||
|     		} | ||||
|     		else if(strcmp(argv[1], "open") == 0) { | ||||
|     		    printf("ATWS: open %s:%d\n", mp3_serv.url, mp3_serv.port); | ||||
|     			connect_close(); | ||||
|     		    return; | ||||
|     		} | ||||
|     		else if(strcmp(argv[1], "close") == 0) { | ||||
|     		    printf("ATWS: close\n"); | ||||
|     			connect_close(); | ||||
|     		    return; | ||||
|     		} | ||||
|     		else if(strcmp(argv[1], "read") == 0) { | ||||
|     			mp3_cfg_read(); | ||||
|     			connect_start(); | ||||
|     		    return; | ||||
|     		} | ||||
|     		else if(strcmp(argv[1], "save") == 0) { | ||||
| 			    printf("ATWS: %s,%d\n", mp3_serv.url, mp3_serv.port); | ||||
|     			if(flash_write_cfg(&mp3_serv, 0x5000, strlen(mp3_serv.port) + strlen(mp3_serv.url))) | ||||
|     			    printf("ATWS: saved\n", mp3_serv.url, mp3_serv.port); | ||||
|     		    return; | ||||
|     		} | ||||
|     	} | ||||
|     	else if (argc >= 3 ) { | ||||
|     		strcpy((char *)mp3_serv.url, (char*)argv[1]); | ||||
|         	mp3_serv.port = atoi((char*)argv[2]); | ||||
|         	printf("ATWS: %s,%d\r\n", mp3_serv.url, mp3_serv.port); | ||||
|         	connect_start(); | ||||
|         	return; | ||||
|     	} | ||||
| 	}	 | ||||
| 	printf("ATWS: Usage: ATWS=URL,PORT or ATWS=close, ATWS=read, ATWS=save\n"); | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void fATWC(void *arg){ | ||||
| 	int mode, ret; | ||||
| 	unsigned long tick1 = xTaskGetTickCount(); | ||||
| 	unsigned long tick2, tick3; | ||||
| 	char empty_bssid[6] = {0}, assoc_by_bssid = 0; | ||||
| 	 | ||||
| 	connect_close(); | ||||
| #if	DEBUG_AT_USER_LEVEL > 1 | ||||
| 	printf("ATWC: Connect to AP...\n"); | ||||
| #endif | ||||
| 	if(memcmp (wifi.bssid.octet, empty_bssid, 6)) | ||||
| 		assoc_by_bssid = 1; | ||||
| 	else if(wifi.ssid.val[0] == 0){ | ||||
| 		printf("ATWC: Error: SSID can't be empty\n"); | ||||
| 		ret = RTW_BADARG; | ||||
| 		goto EXIT; | ||||
| 	} | ||||
| 	if(wifi.password != NULL){ | ||||
| 		if((wifi.key_id >= 0)&&(wifi.key_id <= 3)) { | ||||
| 			wifi.security_type = RTW_SECURITY_WEP_PSK; | ||||
| 		} | ||||
| 		else{ | ||||
| 			wifi.security_type = RTW_SECURITY_WPA2_AES_PSK; | ||||
| 		} | ||||
| 	} | ||||
| 	else{ | ||||
| 		wifi.security_type = RTW_SECURITY_OPEN; | ||||
| 	} | ||||
| 	//Check if in AP mode
 | ||||
| 	wext_get_mode(WLAN0_NAME, &mode); | ||||
| 	if(mode == IW_MODE_MASTER) { | ||||
| 		dhcps_deinit(); | ||||
| 		wifi_off(); | ||||
| 		vTaskDelay(20); | ||||
| 		if (wifi_on(RTW_MODE_STA) < 0){ | ||||
| 			printf("ERROR: Wifi on failed!\n"); | ||||
|                         ret = RTW_ERROR; | ||||
| 			goto EXIT; | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	if(assoc_by_bssid){ | ||||
| 		printf("Joining BSS by BSSID "MAC_FMT" ...\n", MAC_ARG(wifi.bssid.octet)); | ||||
| 		ret = wifi_connect_bssid(wifi.bssid.octet, (char*)wifi.ssid.val, wifi.security_type, (char*)wifi.password,  | ||||
| 						ETH_ALEN, wifi.ssid.len, wifi.password_len, wifi.key_id, NULL);		 | ||||
| 	} else { | ||||
| 		printf("Joining BSS by SSID %s...\n", (char*)wifi.ssid.val); | ||||
| 		ret = wifi_connect((char*)wifi.ssid.val, wifi.security_type, (char*)wifi.password, wifi.ssid.len, | ||||
| 						wifi.password_len, wifi.key_id, NULL); | ||||
| 	} | ||||
| 	 | ||||
| 	if(ret!= RTW_SUCCESS){ | ||||
| 		printf("ERROR: Can't connect to AP\n"); | ||||
| 		goto EXIT; | ||||
| 	} | ||||
| 	tick2 = xTaskGetTickCount(); | ||||
| 	printf("Connected after %dms\n", (tick2-tick1)); | ||||
| 	/* Start DHCPClient */ | ||||
| 	LwIP_DHCP(0, DHCP_START); | ||||
| 	tick3 = xTaskGetTickCount(); | ||||
| 	printf("Got IP after %dms\n", (tick3-tick1)); | ||||
| 	printf("\n\r"); | ||||
| 	connect_start(); | ||||
| EXIT: | ||||
| 	init_wifi_struct( ); | ||||
| } | ||||
| 
 | ||||
| void fATWD(void *arg){ | ||||
| 	int timeout = 20; | ||||
| 	char essid[33]; | ||||
| 	int ret = RTW_SUCCESS; | ||||
| 
 | ||||
| 	connect_close(); | ||||
| #if	DEBUG_AT_USER_LEVEL > 1 | ||||
| 	printf("ATWD: Disconnect...\n"); | ||||
| #endif | ||||
| 	printf("Dissociating AP ...\n"); | ||||
| 	if(wext_get_ssid(WLAN0_NAME, (unsigned char *) essid) < 0) { | ||||
| 		printf("WIFI disconnected\n"); | ||||
| 		goto exit; | ||||
| 	} | ||||
| 
 | ||||
| 	if((ret = wifi_disconnect()) < 0) { | ||||
| 		printf("ERROR: Operation failed!\n"); | ||||
| 		goto exit; | ||||
| 	} | ||||
| 
 | ||||
| 	while(1) { | ||||
| 		if(wext_get_ssid(WLAN0_NAME, (unsigned char *) essid) < 0) { | ||||
| 			printf("WIFI disconnected\n"); | ||||
| 			break; | ||||
| 		} | ||||
| 
 | ||||
| 		if(timeout == 0) { | ||||
| 			printf("ERROR: Deassoc timeout!\n"); | ||||
| 			ret = RTW_TIMEOUT; | ||||
| 			break; | ||||
| 		} | ||||
| 
 | ||||
| 		vTaskDelay(1 * configTICK_RATE_HZ); | ||||
| 		timeout --; | ||||
| 	} | ||||
|     printf("\n\r"); | ||||
| exit: | ||||
|     init_wifi_struct( ); | ||||
| 	return; | ||||
| } | ||||
| 
 | ||||
| // Dump register
 | ||||
| void fATSD(void *arg) | ||||
| { | ||||
| 	int argc = 0; | ||||
| 	char *argv[MAX_ARGC] = {0}; | ||||
| 
 | ||||
| #if	DEBUG_AT_USER_LEVEL > 1 | ||||
| 	printf("ATSD: dump registers\n"); | ||||
| #endif | ||||
| 	if(!arg){ | ||||
| 		printf("ATSD: Usage: ATSD=REGISTER"); | ||||
| 		return; | ||||
| 	} | ||||
| 	argc = parse_param(arg, argv); | ||||
| 	if(argc == 2 || argc == 3) | ||||
| 		CmdDumpWord(argc-1, (unsigned char**)(argv+1)); | ||||
| } | ||||
| 
 | ||||
| void fATSW(void *arg) | ||||
| { | ||||
| 	int argc = 0; | ||||
| 	char *argv[MAX_ARGC] = {0}; | ||||
| 
 | ||||
| #if	DEBUG_AT_USER_LEVEL > 1 | ||||
| 	printf("ATSW: write register\n"); | ||||
| #endif | ||||
| 	if(!arg){ | ||||
| 		printf("ATSW: Usage: ATSW=REGISTER,DATA"); | ||||
| 		return; | ||||
| 	} | ||||
| 	argc = parse_param(arg, argv); | ||||
| 	if(argc == 2 || argc == 3) | ||||
| 		CmdWriteWord(argc-1, (unsigned char**)(argv+1)); | ||||
| } | ||||
| 
 | ||||
| ///// MP3 Set Mode
 | ||||
| // MP3 Off
 | ||||
| void fATOF(void *arg) | ||||
| { | ||||
| #if	DEBUG_AT_USER_LEVEL > 1 | ||||
| 	printf("ATOF: MP3 off...\n"); | ||||
| #endif | ||||
| 	connect_close(); | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void print_wlan_help(void *arg){ | ||||
| 		printf("WLAN AT COMMAND SET:\n"); | ||||
| 		printf("==============================\n"); | ||||
|         printf(" Set MP3 server\n"); | ||||
|         printf("\t# ATWS=URL,PORT\n"); | ||||
|         printf("\tSample:\tATWS=icecast.omroep.nl/3fm-sb-mp3,80\n"); | ||||
|         printf("\t\tATWS=meuk.spritesserver.nl/Ii.Romanzeandante.mp3,80\n"); | ||||
|         printf("\t\tATWS=?, ATWS=close, ATWS=save, ATWS=read\n"); | ||||
|         printf(" Connect to an AES AP\n"); | ||||
|         printf("\t# ATW0=SSID\n"); | ||||
|         printf("\t# ATW1=PASSPHRASE\n"); | ||||
|         printf("\t# ATWC\n"); | ||||
|         printf(" DisConnect AP\n"); | ||||
|         printf("\t# ATWD\n"); | ||||
| } | ||||
| 
 | ||||
| log_item_t at_user_items[ ] = { | ||||
| 	{"ATW0", fATW0,}, | ||||
| 	{"ATW1", fATW1,}, | ||||
| 	{"ATW2", fATW2,}, | ||||
| 	{"ATWC", fATWC,}, | ||||
| 	{"ATST", fATST,}, | ||||
| 	{"ATSD", fATSD,},	// Dump register
 | ||||
| 	{"ATSW", fATSW,},	// Set register
 | ||||
| 	{"ATWD", fATWD,},	//
 | ||||
| 	{"ATWS", fATWS,},	// MP3 Set server, Close connect
 | ||||
| 	{"ATOF", fATOF,},	// MP3 Set Mode
 | ||||
| }; | ||||
| 
 | ||||
| 
 | ||||
| void at_user_init(void) | ||||
| { | ||||
| 	init_wifi_struct(); | ||||
| 	mp3_cfg_read(); | ||||
| 	log_service_add_table(at_user_items, sizeof(at_user_items)/sizeof(at_user_items[0])); | ||||
| } | ||||
| 
 | ||||
| log_module_init(at_user_init); | ||||
| 
 | ||||
| #endif //#ifdef CONFIG_AT_USR
 | ||||
							
								
								
									
										538
									
								
								project/src/user/main.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										538
									
								
								project/src/user/main.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,538 @@ | |||
| /******************************************************************************
 | ||||
|  * | ||||
|  * FileName: user_main.c | ||||
|  * | ||||
|  *******************************************************************************/ | ||||
| #include "rtl8195a/rtl_common.h" | ||||
| #include "rtl8195a.h" | ||||
| #include "hal_log_uart.h" | ||||
| 
 | ||||
| #include "FreeRTOS.h" | ||||
| #include "task.h" | ||||
| //#include "diag.h"
 | ||||
| #include "osdep_service.h" | ||||
| #include "device_lock.h" | ||||
| #include "semphr.h" | ||||
| #include "queue.h" | ||||
| 
 | ||||
| #include <wifi/wifi_conf.h> | ||||
| #include <wifi/wifi_util.h> | ||||
| 
 | ||||
| #include "lwip/sockets.h" | ||||
| #include "lwip/err.h" | ||||
| #include "lwip/dns.h" | ||||
| #include "lwip/netdb.h" | ||||
| #include "dhcp/dhcps.h" | ||||
| 
 | ||||
| #include "mad/mad.h" | ||||
| #include "mad/stream.h" | ||||
| #include "mad/frame.h" | ||||
| #include "mad/synth.h" | ||||
| #include "driver/i2s_freertos.h" | ||||
| #include "user/spiram_fifo.h" | ||||
| #include "user/playerconfig.h" | ||||
| #include "user/atcmd_user.h" | ||||
| #include "main.h" | ||||
| 
 | ||||
| #define DEBUG_MAIN_LEVEL 1 | ||||
| 
 | ||||
| //Priorities of the reader and the decoder thread. Higher = higher prio. (ESP8266!)
 | ||||
| //RTL87xx Higher = lower prio ?
 | ||||
| //#define PRIO_READER (configMAX_PRIORITIES - 2) // (tskIDLE_PRIORITY + PRIORITIE_OFFSET)
 | ||||
| //#define PRIO_MAD (PRIO_READER - 1) // PRIO_READER + n; (TCPIP_THREAD_PRIO = (configMAX_PRIORITIES - 2))
 | ||||
| #define PRIO_MAD (tskIDLE_PRIORITY + 1 + PRIORITIE_OFFSET) | ||||
| #define PRIO_READER (PRIO_MAD + 7) // max 11 ?
 | ||||
| 
 | ||||
| 
 | ||||
| #define mMIN(a, b)  ((a < b)? a : b) | ||||
| 
 | ||||
| //The mp3 read buffer size. 2106 bytes should be enough for up to 48KHz mp3s according to the sox sources. Used by libmad.
 | ||||
| #define READBUFSZ (2106) | ||||
| #define MAX_FIFO_SIZE (16*1024) // min 4*1024 (CPU CLK 166), min 8*1024 (CPU CLK 83MHz), absolute work min = 3*READBUFSZ
 | ||||
| #define MIN_FIFO_HEAP (8*1024) | ||||
| #define SOCK_READ_BUF (256) | ||||
| 
 | ||||
| unsigned char *readBuf; | ||||
| char oversampling = 1; | ||||
| volatile char tskmad_enable, tskreader_enable; | ||||
| static long bufUnderrunCt; | ||||
| 
 | ||||
| // void (*sampToOut)(u32) = i2sPushPWMSamples;
 | ||||
| #define sampToOut i2sPushPWMSamples | ||||
| 
 | ||||
| #ifdef ADD_DEL_SAMPLES // correct smpr
 | ||||
| static char sampCntAdd; | ||||
| static char sampDelCnt; | ||||
| static int sampCnt; | ||||
| #endif | ||||
| 
 | ||||
| 
 | ||||
| // Called by the NXP modifications of libmad. It passes us (for the mono synth)
 | ||||
| // 32 16-bit samples.
 | ||||
| void render_sample_block(short *short_sample_buff, int no_samples) { | ||||
| 	int i; | ||||
| 	for (i = 0; i < no_samples; i++) { | ||||
| 		int x = oversampling; | ||||
| #ifdef ADD_DEL_SAMPLES // correct smpr
 | ||||
| 		if(++sampCnt > 150) { | ||||
| 			sampCnt = 0; | ||||
| 			if (sampDelCnt < 0) { | ||||
| 				//...and don't output an i2s sample
 | ||||
| 				sampDelCnt--; | ||||
| 				x = 0; | ||||
| 			} | ||||
| 			else if (sampDelCnt > 0)  { | ||||
| 				//..and output 2 samples instead of one.
 | ||||
| 				sampDelCnt++; | ||||
| 				x++; | ||||
| 			} | ||||
| 		} | ||||
| #endif | ||||
| 		while(x--) sampToOut((short_sample_buff[i] << 16) | (u16)short_sample_buff[i+no_samples]); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| //Called by the NXP modifications of libmad. Sets the needed output sample rate.
 | ||||
| static int oldRate = 0; | ||||
| void set_dac_sample_rate(int rate, int chls) { | ||||
| 	if (rate == oldRate) return; | ||||
| 	oldRate = rate; | ||||
| #if DEBUG_MAIN_LEVEL > 0 | ||||
| 	DBG_8195A("MAD: Rate %d, channels %d\n", rate, chls); | ||||
| #endif | ||||
| 	oversampling = i2sSetRate(-1, rate); | ||||
| } | ||||
| 
 | ||||
| static enum mad_flow input(struct mad_stream *stream) { | ||||
| 	int n, i; | ||||
| 	int rem; //, fifoLen;
 | ||||
| 	//Shift remaining contents of buf to the front
 | ||||
| 	rem = stream->bufend - stream->next_frame; | ||||
| 	memmove(readBuf, stream->next_frame, rem); | ||||
| 
 | ||||
| 	while (rem < READBUFSZ) { | ||||
| 		n = (READBUFSZ - rem); // Calculate amount of bytes we need to fill buffer.
 | ||||
| 		i = RamFifoFill(); | ||||
| 		if (i < n)	n = i; 		// If the fifo can give us less, only take that amount
 | ||||
| 		if (n == 0) {		// Can't take anything?
 | ||||
| 			// Wait until there is enough data in the buffer. This only happens when the data feed
 | ||||
| 			// rate is too low, and shouldn't normally be needed!
 | ||||
| //			DBG_8195A("Buf uflow, need %d bytes.\n", sizeof(readBuf)-rem);
 | ||||
| 			bufUnderrunCt++; | ||||
| 			// We both silence the output as well as wait a while by pushing silent samples into the i2s system.
 | ||||
| 			// This waits for about 200mS
 | ||||
| #if DEBUG_MAIN_LEVEL > 1 | ||||
| 			DBG_8195A("FIFO: Buffer Underrun\n"); | ||||
| #endif | ||||
| 			for (n = 0; n < 441*2; n++)	sampToOut(0); | ||||
| 		} else { | ||||
| 			//Read some bytes from the FIFO to re-fill the buffer.
 | ||||
| 			RamFifoRead(&readBuf[rem], n); | ||||
| 			rem += n; | ||||
| 		} | ||||
| #ifdef ADD_DEL_SAMPLES | ||||
| 		if(i < READBUFSZ) { | ||||
| 			sampCntAdd = 10; // add samples
 | ||||
| 		} | ||||
| 		else if(RamFifoLen() - i < SOCK_READ_BUF) { // fifo free < SOCK_READ_BUF
 | ||||
| 
 | ||||
| 			sampCntAdd = -1; // del samples
 | ||||
| 		} | ||||
| 		else { | ||||
| 				sampCntAdd++; // add samples
 | ||||
| 		} | ||||
| 		sampDelCnt += sampCntAdd; | ||||
| #endif | ||||
| 	} | ||||
| 	//Okay, let MAD decode the buffer.
 | ||||
| 	mad_stream_buffer(stream, readBuf, READBUFSZ); | ||||
| 	return MAD_FLOW_CONTINUE; | ||||
| } | ||||
| 
 | ||||
| //Routine to print out an error
 | ||||
| static enum mad_flow error(void *data, struct mad_stream *stream, | ||||
| 		struct mad_frame *frame) { | ||||
| #if DEBUG_MAIN_LEVEL > 0 | ||||
| 	DBG_8195A("MAD: Dec err 0x%04x (%s)\n", stream->error, | ||||
| 			mad_stream_errorstr(stream)); | ||||
| #endif | ||||
| 	return MAD_FLOW_CONTINUE; | ||||
| } | ||||
| 
 | ||||
| void tskreader(void *pvParameters); | ||||
| 
 | ||||
| //This is the main mp3 decoding task. It will grab data from the input buffer FIFO in the SPI ram and
 | ||||
| //output it to the I2S port.
 | ||||
| void tskmad(void *pvParameters) { | ||||
| 	//Initialize I2S
 | ||||
| 	if (i2sInit(-1, I2S_DMA_PAGE_WAIT_MS_MIN * I2S_DMA_PAGE_SIZE_MS_96K, WL_24b)) { // min 2 ms x I2S_DMA_PAGE_SIZE buffers
 | ||||
| 		//Allocate structs needed for mp3 decoding
 | ||||
| 		char * mad_bufs = pvPortMalloc( | ||||
| 				sizeof(struct mad_stream) + sizeof(struct mad_frame) | ||||
| 						+ sizeof(struct mad_synth) + READBUFSZ); | ||||
| 		if (mad_bufs == NULL) { | ||||
| #if DEBUG_MAIN_LEVEL > 0 | ||||
| 			DBG_8195A("MAD: Alloc failed\n"); | ||||
| #endif | ||||
| 			goto exit; | ||||
| 		} | ||||
| 		rtl_memset(mad_bufs, 0, | ||||
| 				sizeof(struct mad_stream) + sizeof(struct mad_frame) | ||||
| 						+ sizeof(struct mad_synth)); | ||||
| #if DEBUG_MAIN_LEVEL > 0 | ||||
| 		DBG_8195A("MAD: Alloc %d bytes at %p\n", | ||||
| 				sizeof(struct mad_stream) + sizeof(struct mad_frame) + sizeof(struct mad_synth) + READBUFSZ, | ||||
| 				mad_bufs); | ||||
| #endif | ||||
| 		struct mad_stream *stream = mad_bufs; | ||||
| 		struct mad_frame *frame = &mad_bufs[sizeof(struct mad_stream)]; | ||||
| 		struct mad_synth *synth = &mad_bufs[sizeof(struct mad_stream) | ||||
| 				+ sizeof(struct mad_frame)]; | ||||
| 		readBuf = &mad_bufs[sizeof(struct mad_stream) + sizeof(struct mad_frame) | ||||
| 				+ sizeof(struct mad_synth)]; | ||||
| 
 | ||||
| 		bufUnderrunCt = 0; | ||||
| 		oldRate = 0; | ||||
| 		oversampling = 1; | ||||
| #ifdef ADD_DEL_SAMPLES | ||||
| 		sampCntAdd = 0; | ||||
| 		sampCnt = 0; | ||||
| 		sampDelCnt = 0; | ||||
| #endif | ||||
| 		//Initialize mp3 parts
 | ||||
| #if DEBUG_MAIN_LEVEL > 0 | ||||
| 		DBG_8195A("MAD: Decoder start.\n"); | ||||
| #endif | ||||
| 		mad_stream_init(stream); | ||||
| 		mad_frame_init(frame); | ||||
| 		mad_synth_init(synth); | ||||
| 		while (tskmad_enable == 1) { | ||||
| 			input(stream); //calls mad_stream_buffer internally
 | ||||
| 			while (tskmad_enable == 1) { | ||||
| #if DEBUG_MAIN_LEVEL > 3 | ||||
| 				DBG_8195A("MAD: Frame decode.\n"); | ||||
| #endif | ||||
| 				int r = mad_frame_decode(frame, stream); | ||||
| 				if (r == -1) { | ||||
| #if DEBUG_MAIN_LEVEL > 2 | ||||
| 					 DBG_8195A("MAD: Frame error.\n"); | ||||
| #endif | ||||
| 					if (!MAD_RECOVERABLE(stream->error)) { | ||||
| 						//We're most likely out of buffer and need to call input() again
 | ||||
| 						break; | ||||
| 					} | ||||
| 					error(NULL, stream, frame); | ||||
| 					continue; | ||||
| 				} | ||||
| #if DEBUG_MAIN_LEVEL > 3 | ||||
| 				DBG_8195A("MAD: Frame synth.\n"); | ||||
| #endif | ||||
| 				mad_synth_frame(synth, frame); | ||||
| 			} | ||||
| 		}; | ||||
| 		mad_synth_finish(synth); | ||||
| 		mad_frame_finish(frame); | ||||
| 		mad_stream_finish(stream); | ||||
| 		vTaskDelay(10); | ||||
| 		vPortFree(mad_bufs); | ||||
| 	} | ||||
| #if DEBUG_MAIN_LEVEL > 0 | ||||
| 	DBG_8195A("MAD: Closed.\n"); | ||||
| #endif | ||||
| exit: | ||||
| 	i2sClose(-1); | ||||
| 	tskreader_enable = 0; | ||||
| 	tskmad_enable = -1; | ||||
| 	vTaskDelete(NULL); | ||||
| } | ||||
| 
 | ||||
| int getIpForHost(const char *host, struct sockaddr_in *ip) { | ||||
| 	struct hostent *he; | ||||
| 	struct in_addr **addr_list; | ||||
| 	he = gethostbyname(host); | ||||
| 	if (he == NULL) return 0; | ||||
| 	addr_list = (struct in_addr **) he->h_addr_list; | ||||
| 	if (addr_list[0] == NULL) return 0; | ||||
| 	ip->sin_family = AF_INET; | ||||
| 	memcpy(&ip->sin_addr, addr_list[0], sizeof(ip->sin_addr)); | ||||
| 	return 1; | ||||
| } | ||||
| 
 | ||||
| //Open a connection to a webserver and request an URL. Yes, this possibly is one of the worst ways to do this,
 | ||||
| //but RAM is at a premium here, and this works for most of the cases.
 | ||||
| int openConn(const char *streamHost, const char *streamPath, int streamPort) { | ||||
| 	int n = 5; | ||||
| 	while (tskreader_enable == 1) { | ||||
| 		struct sockaddr_in remote_ip; | ||||
| 		bzero(&remote_ip, sizeof(struct sockaddr_in)); | ||||
| 		if (!getIpForHost(streamHost, &remote_ip)) { | ||||
| 			vTaskDelay(1000 / portTICK_RATE_MS); | ||||
| 			if(n--)	continue; | ||||
| #if DEBUG_MAIN_LEVEL > 0 | ||||
| 			DBG_8195A("MP3: Not get IP server <%s>!\n", streamHost); | ||||
| #endif | ||||
| 			return -1; | ||||
| 		} | ||||
| 		int sock = socket(PF_INET, SOCK_STREAM, 0); | ||||
| 		if (sock == -1) { | ||||
| #if DEBUG_MAIN_LEVEL > 0 | ||||
| 			DBG_8195A("MP3: Not open socket!\n"); | ||||
| #endif | ||||
| //			tskreader_enable = 0;
 | ||||
| 			return -1; | ||||
| 		} | ||||
| 
 | ||||
| 		remote_ip.sin_port = htons(streamPort); | ||||
| #if DEBUG_MAIN_LEVEL > 0 | ||||
| 		DBG_8195A("MP3: Connecting to server %s...\n", | ||||
| 				ipaddr_ntoa((const ip_addr_t* )&remote_ip.sin_addr.s_addr)); | ||||
| #endif | ||||
| 		if (connect(sock, (struct sockaddr * )(&remote_ip), | ||||
| 				sizeof(struct sockaddr)) != 00) { | ||||
| 			close(sock); | ||||
| #if DEBUG_MAIN_LEVEL > 0 | ||||
| 			DBG_8195A("MP3: Connect error!\n"); | ||||
| #endif | ||||
| //			vTaskDelay(1000 / portTICK_RATE_MS);
 | ||||
| //			continue;
 | ||||
| 			return -1; | ||||
| 		} | ||||
| 		//Cobble together HTTP request
 | ||||
| 		write(sock, "GET ", 4); | ||||
| 		write(sock, streamPath, strlen(streamPath)); | ||||
| 		write(sock, " HTTP/1.0\r\nHost: ", 17); | ||||
| 		write(sock, streamHost, strlen(streamHost)); | ||||
| 		write(sock, "\r\n\r\n", 4); | ||||
| 		//We ignore the headers that the server sends back... it's pretty dirty in general to do that,
 | ||||
| 		//but it works here because the MP3 decoder skips it because it isn't valid MP3 data.
 | ||||
| 		return sock; | ||||
| 	} | ||||
| 	return -1; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| int http_head_read(unsigned char *buf, int len, int ff) { | ||||
| 	int flg_head = 0; | ||||
| 	int n, ret = 0; | ||||
| 	if ((n = read(ff, buf, len)) <= 0)	return 0; | ||||
| 	if(n > 11 && *((u32 *)buf) == 0x50545448) { // "HTTP" // HTTP/1.0 200 OK
 | ||||
| 		int x; | ||||
| 		for(x = 3; x < n && buf[x] != ' '; x++); | ||||
| 		while(x < n && buf[x] == ' ') x++; | ||||
| 		if(x < n) ret = atoi(&buf[x]); | ||||
| 		int cnt = 0; | ||||
| 		x = 0; | ||||
| 		while(ret) { | ||||
| 			int z = 0; | ||||
| 			while (x < n) { | ||||
| 				if (cnt++ > 16384)	return 600; // Header Too Large
 | ||||
| 				if (buf[x++] == ((flg_head & 1) ? 0x0a : 0x0d)) { | ||||
| 					if ((flg_head & 3) == 1) { | ||||
| #if DEBUG_MAIN_LEVEL > 0 | ||||
| 						buf[x-1] = 0; | ||||
| 						DBG_8195A("%s\n", &buf[z]); | ||||
| #endif | ||||
| 						z = x; | ||||
| 					} | ||||
| 					if (flg_head >= 3) { | ||||
| 						if (n - x > 0) RamFifoWrite(&buf[x], n - x); | ||||
| #if DEBUG_MAIN_LEVEL > 1 | ||||
| 						DBG_8195A("MP3: Skip HTTP head in %d bytes\n\n", cnt); | ||||
| #endif | ||||
| 						return ret; | ||||
| 					} | ||||
| 					flg_head++; | ||||
| 				} | ||||
| 				else flg_head = 0; | ||||
| 			} | ||||
| 			x = 0; | ||||
| 			while(z < n) buf[x++] = buf[z++]; | ||||
| 			if ((n = read(ff, &buf[x], len - x)) <= 0) return 601; // content ??
 | ||||
| 			n += x; | ||||
| 		}; | ||||
| 	} | ||||
| 	else RamFifoWrite(buf, n); | ||||
| 	return ret; | ||||
| } | ||||
| 
 | ||||
| //Reader task. This will try to read data from a TCP socket into the SPI fifo buffer.
 | ||||
| void tskreader(void *pvParameters) { | ||||
| 	char wbuf[SOCK_READ_BUF]; | ||||
| 	int n; | ||||
| 	if (RamFifoInit(mMIN(xPortGetFreeHeapSize() - MIN_FIFO_HEAP, MAX_FIFO_SIZE))) { | ||||
| #if I2S_DEBUG_LEVEL > 1 | ||||
| 		unsigned int t = xTaskGetTickCount(); | ||||
| #endif | ||||
| 		while (tskreader_enable == 1) { | ||||
| 			n = strlen(mp3_serv.url); | ||||
| 			int i; | ||||
| 			u8 * uri = NULL; | ||||
| 			for(i = 0; i < n; i++) { | ||||
| 				wbuf[i] = mp3_serv.url[i]; | ||||
| 				if(wbuf[i] == '/') { | ||||
| 					wbuf[i] = 0; | ||||
| 					uri = &mp3_serv.url[i]; | ||||
| 					break; | ||||
| 				} | ||||
| 			} | ||||
| 			if(uri == NULL) { | ||||
| #if DEBUG_MAIN_LEVEL > 0 | ||||
| 				DBG_8195A("MP3: Error url <%s>!\n", mp3_serv.url); | ||||
| #endif | ||||
| 				tskreader_enable = 0; | ||||
| 				break; | ||||
| 			} | ||||
| 			int fd = openConn(wbuf, uri, mp3_serv.port); | ||||
| 			if(fd < 0) { | ||||
| 				tskreader_enable = 0; | ||||
| 				break; | ||||
| 			} | ||||
| 			if ((n = http_head_read(wbuf, sizeof(wbuf), fd)) != 200) { | ||||
| #if DEBUG_MAIN_LEVEL > 0 | ||||
| 				DBG_8195A("MP3: HTTP error %d\n", n); | ||||
| #endif | ||||
| 				tskreader_enable = 0; | ||||
| 				break; | ||||
| 			} | ||||
| 			else do { | ||||
| 				n = read(fd, wbuf, sizeof(wbuf)); | ||||
| 				// DBG_8195A("Socket read %d bytes\n", n);
 | ||||
| 				if (n > 0)	RamFifoWrite(wbuf, n); | ||||
| 				if ((tskmad_enable != 1) && (RamFifoFree() < RamFifoLen() / 2)) { | ||||
| #if DEBUG_MAIN_LEVEL > 0 | ||||
| 					DBG_8195A("FIFO: Start Buffer fill %d\n", RamFifoFill()); | ||||
| #endif | ||||
| 					// Buffer is filled. Start up the MAD task. Yes, the 2100 words of stack is a fairly large amount but MAD seems to need it.
 | ||||
| 					tskmad_enable = 1; | ||||
| 					if (xTaskCreate(tskmad, "tskmad", 2100, NULL, PRIO_MAD,	NULL) != pdPASS) { | ||||
| #if DEBUG_MAIN_LEVEL > 0 | ||||
| 						DBG_8195A("MP3: Error creating MAD task! Out of memory?\n"); | ||||
| #endif | ||||
| 						tskmad_enable = 0; | ||||
| 						tskreader_enable = 0; | ||||
| 						break; | ||||
| 					} | ||||
| 				} | ||||
| #if I2S_DEBUG_LEVEL > 1 | ||||
| 				if (xTaskGetTickCount() - t > 3000) { | ||||
| 					t = xTaskGetTickCount(); | ||||
| 					DBG_8195A("MP3: Buffer fill %d, DMA underrun ct %d, buff underrun ct %d\n", RamFifoFill(), (int )i2sGetUnderrunCnt(), bufUnderrunCt); | ||||
| 				} | ||||
| #endif | ||||
| 			} while (n > 0 && (tskreader_enable == 1)); | ||||
| 			if(fd >= 0) { | ||||
| #if DEBUG_MAIN_LEVEL > 1 | ||||
| 				if(n == 0) { | ||||
| 					u32 err; | ||||
| 					socklen_t slen = sizeof(err); | ||||
| 					if(!lwip_getsockopt(fd, SOL_SOCKET, SO_ERROR, &err, &slen)) { | ||||
| 						DBG_8195A("MP3: Socket error %d\n", err); | ||||
| 					} | ||||
| 				} | ||||
| #endif | ||||
| 				close(fd); | ||||
| 			} | ||||
| 		} | ||||
| #if DEBUG_MAIN_LEVEL > 0 | ||||
| 		DBG_8195A("MP3: Connection closed.\n"); | ||||
| #endif | ||||
| 	} | ||||
| 	if(tskmad_enable == 1) { | ||||
| 		tskmad_enable = 0; | ||||
| 		while (tskmad_enable == 0) vTaskDelay(2); | ||||
| 	} | ||||
| 	RamFifoClose(); | ||||
| #if DEBUG_MAIN_LEVEL > 2 | ||||
| 	DBG_8195A("\nMP3: Task reader closed.\n"); | ||||
| #endif | ||||
| 	tskreader_enable = -1; | ||||
| 	vTaskDelete(NULL); | ||||
| } | ||||
| 
 | ||||
| //We need this to tell the OS we're running at a higher clock frequency.
 | ||||
| //sk//extern void os_update_cpu_frequency(int mhz);
 | ||||
| 
 | ||||
| void connect_close(void) { | ||||
| 	if (tskreader_enable == 1) { | ||||
| 		tskreader_enable = 0; | ||||
| 		while(tskreader_enable != -1) vTaskDelay(2); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| void connect_start(void) { | ||||
| 	connect_close(); | ||||
| 	if(mp3_serv.port != 0 && strlen(mp3_serv.url) > 2) { | ||||
| #if DEBUG_MAIN_LEVEL > 0 | ||||
| 		DBG_8195A("MP3: Connect url: %s:%d\n", mp3_serv.url, mp3_serv.port); | ||||
| //		DBG_8195A("Waiting for network.\n");
 | ||||
| #endif | ||||
| 		//Fire up the reader task. The reader task will fire up the MP3 decoder as soon
 | ||||
| 		//as it has read enough MP3 data.
 | ||||
| 		tskreader_enable = 1; | ||||
| 		if (xTaskCreate(tskreader, "tskreader", 300, NULL, PRIO_READER,	NULL) != pdPASS) { | ||||
| #if DEBUG_MAIN_LEVEL > 0 | ||||
| 			DBG_8195A("\n\r%s xTaskCreate(tskreader) failed", __FUNCTION__); | ||||
| #endif | ||||
| 			tskreader_enable = 0; | ||||
| 		} | ||||
| 	} | ||||
| #if DEBUG_MAIN_LEVEL > 0 | ||||
| 	else { | ||||
| 		DBG_8195A("MP3: No set url!\n"); | ||||
| 	} | ||||
| #endif | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| /**
 | ||||
|  * @brief  Main program. | ||||
|  * @param  None | ||||
|  * @retval None | ||||
|  */ | ||||
| 
 | ||||
| void main(void) { | ||||
| #if DEBUG_MAIN_LEVEL > 2 | ||||
| 	 ConfigDebugErr  = -1; | ||||
| 	 ConfigDebugInfo = -1; | ||||
| 	 ConfigDebugWarn = -1; | ||||
| #endif | ||||
| /*
 | ||||
| 	 if ( rtl_cryptoEngine_init() != 0 ) DBG_8195A("crypto engine init failed\r\n"); | ||||
| */ | ||||
| #if defined(CONFIG_CPU_CLK) | ||||
| 		HalCpuClkConfig(CPU_CLOCK_SEL_VALUE); // 0 - 166666666 Hz, 1 - 83333333 Hz, 2 - 41666666 Hz, 3 - 20833333 Hz, 4 - 10416666 Hz, 5 - 4000000 Hz
 | ||||
| 		HAL_LOG_UART_ADAPTER pUartAdapter; | ||||
| 		pUartAdapter.BaudRate = RUART_BAUD_RATE_38400; | ||||
| 		HalLogUartSetBaudRate(&pUartAdapter); | ||||
| 		SystemCoreClockUpdate(); | ||||
| 		En32KCalibration(); | ||||
| #endif | ||||
| #if DEBUG_MAIN_LEVEL > 1 | ||||
| 	DBG_INFO_MSG_ON(_DBG_TCM_HEAP_); // On Debug TCM MEM
 | ||||
| #endif | ||||
| #if DEBUG_MAIN_LEVEL > 0 | ||||
| 	vPortFree(pvPortMalloc(4)); // Init RAM heap 
 | ||||
| 	fATST(NULL); // RAM/TCM/Heaps info
 | ||||
| #endif | ||||
| 	/* pre-processor of application example */ | ||||
| 	pre_example_entry(); | ||||
| 
 | ||||
| 	/* wlan intialization */ | ||||
| #if defined(CONFIG_WIFI_NORMAL) && defined(CONFIG_NETWORK) | ||||
| 	wlan_network(); | ||||
| #endif | ||||
| 	/* Initialize log uart and at command service */ | ||||
| 	console_init(); | ||||
| 
 | ||||
| 	/* Execute application example */ | ||||
| 	example_entry(); | ||||
| 
 | ||||
| 	/*Enable Schedule, Start Kernel*/ | ||||
| #if defined(CONFIG_KERNEL) && !TASK_SCHEDULER_DISABLED | ||||
| #ifdef PLATFORM_FREERTOS | ||||
| 	vTaskStartScheduler(); | ||||
| #endif | ||||
| #else | ||||
| 	RtlConsolTaskRom(NULL); | ||||
| #endif | ||||
| } | ||||
							
								
								
									
										181
									
								
								project/src/user/spiram_fifo.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										181
									
								
								project/src/user/spiram_fifo.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,181 @@ | |||
| /******************************************************************************
 | ||||
|  * | ||||
|  * FileName: spiram_fifo.c | ||||
|  * | ||||
| *******************************************************************************/ | ||||
| #include "rtl8195a/rtl_common.h" | ||||
| #include "diag.h" | ||||
| 
 | ||||
| #include "FreeRTOS.h" | ||||
| #include "task.h" | ||||
| #include "semphr.h" | ||||
| #include "queue.h" | ||||
| 
 | ||||
| #include "user/spiram_fifo.h" | ||||
| #include "user/playerconfig.h" | ||||
| 
 | ||||
| 
 | ||||
| typedef struct _sBUF_FIFO_ { | ||||
| 	xSemaphoreHandle mux; | ||||
| 	xSemaphoreHandle semCanRead; | ||||
| 	xSemaphoreHandle semCanWrite; | ||||
| 	int fifoRpos, fifoWpos, fifoFill, fifoSize; | ||||
| 	long fifoOvfCnt, fifoUdrCnt; | ||||
| 	unsigned char * buf; | ||||
| } BUF_FIFO, * PBUF_FIFO; | ||||
| 
 | ||||
| PBUF_FIFO pbuf_fifo; | ||||
| 
 | ||||
| #define FIFO_REZSIZE 2048 | ||||
| 
 | ||||
| void RamFifoClose(void) { | ||||
| 	if(pbuf_fifo != NULL) { | ||||
| 		if(pbuf_fifo->mux != NULL) vSemaphoreDelete(pbuf_fifo->mux); // xSemaphoreTake(mux, portMAX_DELAY);
 | ||||
| 		if(pbuf_fifo->semCanRead != NULL) vSemaphoreDelete(pbuf_fifo->semCanRead); | ||||
| 		if(pbuf_fifo->semCanWrite != NULL) vSemaphoreDelete(pbuf_fifo->semCanWrite); | ||||
| 		if(pbuf_fifo->buf != NULL) vPortFree(pbuf_fifo->buf); | ||||
| 		vPortFree(pbuf_fifo); | ||||
| 		pbuf_fifo = NULL; | ||||
| 		DBG_8195A("FIFO: Closed.\n"); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| static int RamFifoAlloc(int size) { | ||||
| 	pbuf_fifo = (PBUF_FIFO) pvPortMalloc(sizeof(BUF_FIFO)); | ||||
| 	if(pbuf_fifo == NULL) return 0; | ||||
| 	pbuf_fifo->mux = NULL; | ||||
| 	pbuf_fifo->semCanRead = NULL; | ||||
| 	pbuf_fifo->semCanWrite = NULL; | ||||
| 	pbuf_fifo->fifoSize = 0; | ||||
| 	pbuf_fifo->buf = pvPortMalloc(size); | ||||
| 	if(pbuf_fifo->buf == NULL) return 0; | ||||
| 	pbuf_fifo->fifoSize = size; | ||||
| 	vSemaphoreCreateBinary(pbuf_fifo->semCanRead); | ||||
| 	if(pbuf_fifo->semCanRead == NULL) return 0; | ||||
| 	vSemaphoreCreateBinary(pbuf_fifo->semCanWrite); | ||||
| 	if(pbuf_fifo->semCanWrite == NULL) return 0; | ||||
| 	pbuf_fifo->mux = xSemaphoreCreateMutex(); | ||||
| 	if(pbuf_fifo->mux == NULL) return 0; | ||||
| 	return 1; | ||||
| } | ||||
| 
 | ||||
| //Initialize the FIFO
 | ||||
| int RamFifoInit(int size) { | ||||
| 	if(size < 2*FIFO_REZSIZE) { | ||||
| 		DBG_8195A("FIFO: Buffer size < %d?", 2*FIFO_REZSIZE); | ||||
| 		return 0; | ||||
| 	} | ||||
| 	if(pbuf_fifo == NULL) { | ||||
| 		if (!RamFifoAlloc(size)) { | ||||
| 			RamFifoClose(); | ||||
| 			DBG_8195A("FIFO: Low Heap!\n"); | ||||
| 			return 0; | ||||
| 		} | ||||
| 	} | ||||
| 	xSemaphoreTake(pbuf_fifo->mux, portMAX_DELAY); | ||||
| 	pbuf_fifo->fifoRpos = 0; | ||||
| 	pbuf_fifo->fifoWpos = 0; | ||||
| 	pbuf_fifo->fifoFill = 0; | ||||
| 	pbuf_fifo->fifoOvfCnt = 0; | ||||
| 	pbuf_fifo->fifoUdrCnt = 0; | ||||
| 	if (pbuf_fifo->fifoSize != size) { | ||||
| 		vPortFree(pbuf_fifo->buf); | ||||
| 		pbuf_fifo->buf = pvPortMalloc(size); | ||||
| 		if(pbuf_fifo->buf == NULL) { | ||||
| 			pbuf_fifo->fifoSize = 0; | ||||
| 			xSemaphoreGive(pbuf_fifo->mux); | ||||
| 			DBG_8195A("FIFO: Low Heap!\n"); | ||||
| 			return 0; | ||||
| 		} | ||||
| 		pbuf_fifo->fifoSize = size; | ||||
| 	} | ||||
| 	DBG_8195A("FIFO: Alloc %d bytes at %p\n", pbuf_fifo->fifoSize, pbuf_fifo->buf); | ||||
| 	xSemaphoreGive(pbuf_fifo->mux); | ||||
| 	return 1; | ||||
| } | ||||
| 
 | ||||
| // Read bytes from the FIFO
 | ||||
| void RamFifoRead(char *buff, int len) { | ||||
| 	while (len>0) { | ||||
| 		int n = len; | ||||
| //		if (n > FIFO_REZSIZE) n = FIFO_REZSIZE;			//don't read more than SPIREADSIZE
 | ||||
| 		if (n > (pbuf_fifo->fifoSize - pbuf_fifo->fifoRpos)) n = pbuf_fifo->fifoSize - pbuf_fifo->fifoRpos; //don't read past end of buffer
 | ||||
| 		xSemaphoreTake(pbuf_fifo->mux, portMAX_DELAY); | ||||
| 		if (pbuf_fifo->fifoFill < n) { | ||||
| 			// DBG_8195A("FIFO empty.\n");
 | ||||
| 			//Drat, not enough data in FIFO. Wait till there's some written and try again.
 | ||||
| 			pbuf_fifo->fifoUdrCnt++; | ||||
| 			xSemaphoreGive(pbuf_fifo->mux); | ||||
| 			if (pbuf_fifo->fifoFill < pbuf_fifo->fifoSize - FIFO_REZSIZE) xSemaphoreTake(pbuf_fifo->semCanRead, portMAX_DELAY); | ||||
| 		} else { | ||||
| 			//Read the data.
 | ||||
| 			memcpy(buff, &pbuf_fifo->buf[pbuf_fifo->fifoRpos], n); | ||||
| 			buff += n; | ||||
| 			len -= n; | ||||
| 			pbuf_fifo->fifoFill -= n; | ||||
| 			pbuf_fifo->fifoRpos += n; | ||||
| 			if (pbuf_fifo->fifoRpos >= pbuf_fifo->fifoSize) pbuf_fifo->fifoRpos = 0; | ||||
| 			xSemaphoreGive(pbuf_fifo->mux); | ||||
| 			xSemaphoreGive(pbuf_fifo->semCanWrite); //Indicate writer thread there's some free room in the fifo
 | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| //Write bytes to the FIFO
 | ||||
| void RamFifoWrite(char *buff, int len) { | ||||
| 	while (len > 0) { | ||||
| 		int n = len; | ||||
| //		if (n > FIFO_REZSIZE) n = FIFO_REZSIZE;		//don't read more than SPIREADSIZE
 | ||||
| 		if (n > (pbuf_fifo->fifoSize - pbuf_fifo->fifoWpos)) n = pbuf_fifo->fifoSize - pbuf_fifo->fifoWpos; //don't read past end of buffer
 | ||||
| 
 | ||||
| 		xSemaphoreTake(pbuf_fifo->mux, portMAX_DELAY); | ||||
| 		if ((pbuf_fifo->fifoSize - pbuf_fifo->fifoFill) < n) { | ||||
| 			// DBG_8195A("FIFO full.\n");
 | ||||
| 			//Drat, not enough free room in FIFO. Wait till there's some read and try again.
 | ||||
| 			pbuf_fifo->fifoOvfCnt++; | ||||
| 			xSemaphoreGive(pbuf_fifo->mux); | ||||
| 			xSemaphoreTake(pbuf_fifo->semCanWrite, portMAX_DELAY); | ||||
| 		} else { | ||||
| 			// Write the data.
 | ||||
| 			memcpy(&pbuf_fifo->buf[pbuf_fifo->fifoWpos], buff, n); | ||||
| 			buff += n; | ||||
| 			len -= n; | ||||
| 			pbuf_fifo->fifoFill += n; | ||||
| 			pbuf_fifo->fifoWpos += n; | ||||
| 			if (pbuf_fifo->fifoWpos >= pbuf_fifo->fifoSize) pbuf_fifo->fifoWpos = 0; | ||||
| 			xSemaphoreGive(pbuf_fifo->mux); | ||||
| 			xSemaphoreGive(pbuf_fifo->semCanRead); //Tell reader thread there's some data in the fifo.
 | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| //Get amount of bytes in use
 | ||||
| int RamFifoFill() { | ||||
| 	xSemaphoreTake(pbuf_fifo->mux, portMAX_DELAY); | ||||
| 	int ret = pbuf_fifo->fifoFill; | ||||
| 	xSemaphoreGive(pbuf_fifo->mux); | ||||
| 	return ret; | ||||
| } | ||||
| 
 | ||||
| int RamFifoFree() { | ||||
| 	return (pbuf_fifo->fifoSize - RamFifoFill()); | ||||
| } | ||||
| 
 | ||||
| int RamFifoLen() { | ||||
| 	return pbuf_fifo->fifoSize; | ||||
| } | ||||
| 
 | ||||
| long RamGetOverrunCt() { | ||||
| 	xSemaphoreTake(pbuf_fifo->mux, portMAX_DELAY); | ||||
| 	long ret = pbuf_fifo->fifoOvfCnt; | ||||
| 	xSemaphoreGive(pbuf_fifo->mux); | ||||
| 	return ret; | ||||
| } | ||||
| 
 | ||||
| long RamGetUnderrunCt() { | ||||
| 	xSemaphoreTake(pbuf_fifo->mux, portMAX_DELAY); | ||||
| 	long ret = pbuf_fifo->fifoUdrCnt; | ||||
| 	xSemaphoreGive(pbuf_fifo->mux); | ||||
| 	return ret; | ||||
| } | ||||
| 
 | ||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue