forked from len0rd/rockbox
		
	git-svn-id: svn://svn.rockbox.org/rockbox/trunk@15977 a1c6a512-1295-4272-9138-f99709370657
		
			
				
	
	
		
			262 lines
		
	
	
	
		
			6.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			262 lines
		
	
	
	
		
			6.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * alloc.c
 | |
|  * Copyright (C) 2000-2003 Michel Lespinasse <walken@zoy.org>
 | |
|  * Copyright (C) 1999-2000 Aaron Holtzman <aholtzma@ess.engr.uvic.ca>
 | |
|  *
 | |
|  * This file is part of mpeg2dec, a free MPEG-2 video stream decoder.
 | |
|  * See http://libmpeg2.sourceforge.net/ for updates.
 | |
|  *
 | |
|  * mpeg2dec 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.
 | |
|  *
 | |
|  * mpeg2dec 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
 | |
|  */
 | |
| 
 | |
| #include "plugin.h"
 | |
| #include "mpegplayer.h"
 | |
| 
 | |
| /* Main allocator */
 | |
| static off_t mem_ptr;
 | |
| static size_t bufsize;
 | |
| static unsigned char* mallocbuf;
 | |
| 
 | |
| /* libmpeg2 allocator */
 | |
| static off_t mpeg2_mem_ptr NOCACHEBSS_ATTR;
 | |
| static size_t mpeg2_bufsize NOCACHEBSS_ATTR;
 | |
| static unsigned char *mpeg2_mallocbuf NOCACHEBSS_ATTR;
 | |
| static unsigned char *mpeg2_bufallocbuf NOCACHEBSS_ATTR;
 | |
| 
 | |
| #if defined(DEBUG) || defined(SIMULATOR)
 | |
| const char * mpeg_get_reason_str(int reason)
 | |
| {
 | |
|     const char *str;
 | |
| 
 | |
|     switch (reason)
 | |
|     {
 | |
|     case MPEG2_ALLOC_MPEG2DEC:
 | |
|         str = "MPEG2_ALLOC_MPEG2DEC";
 | |
|         break;
 | |
|     case MPEG2_ALLOC_CHUNK:
 | |
|         str = "MPEG2_ALLOC_CHUNK";
 | |
|         break;
 | |
|     case MPEG2_ALLOC_YUV:
 | |
|         str = "MPEG2_ALLOC_YUV";
 | |
|         break;
 | |
|     case MPEG2_ALLOC_CONVERT_ID:
 | |
|         str = "MPEG2_ALLOC_CONVERT_ID";
 | |
|         break;
 | |
|     case MPEG2_ALLOC_CONVERTED:
 | |
|         str = "MPEG2_ALLOC_CONVERTED";
 | |
|         break;
 | |
|     case MPEG_ALLOC_MPEG2_BUFFER:
 | |
|         str = "MPEG_ALLOC_MPEG2_BUFFER";
 | |
|         break;
 | |
|     case MPEG_ALLOC_AUDIOBUF:
 | |
|         str = "MPEG_ALLOC_AUDIOBUF";
 | |
|         break;
 | |
|     case MPEG_ALLOC_PCMOUT:
 | |
|         str = "MPEG_ALLOC_PCMOUT";
 | |
|         break;
 | |
|     case MPEG_ALLOC_DISKBUF:
 | |
|         str = "MPEG_ALLOC_DISKBUF";
 | |
|         break;
 | |
|     case MPEG_ALLOC_CODEC_MALLOC:
 | |
|         str = "MPEG_ALLOC_CODEC_MALLOC";
 | |
|         break;
 | |
|     case MPEG_ALLOC_CODEC_CALLOC:
 | |
|         str = "MPEG_ALLOC_CODEC_CALLOC";
 | |
|         break;
 | |
|     default:
 | |
|         str = "Unknown";
 | |
|     }
 | |
| 
 | |
|     return str;        
 | |
| }
 | |
| #endif
 | |
| 
 | |
| static void * mpeg_malloc_internal (unsigned char *mallocbuf,
 | |
|                                     off_t *mem_ptr,
 | |
|                                     size_t bufsize,
 | |
|                                     unsigned size,
 | |
|                                     int reason)
 | |
| {
 | |
|     void *x;
 | |
| 
 | |
|     DEBUGF("mpeg_alloc_internal: bs:%lu s:%u reason:%s (%d)\n",
 | |
|            bufsize, size, mpeg_get_reason_str(reason), reason);
 | |
| 
 | |
|     if ((size_t) (*mem_ptr + size) > bufsize)
 | |
|     {
 | |
|         DEBUGF("OUT OF MEMORY\n");
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     x = &mallocbuf[*mem_ptr];
 | |
|     *mem_ptr += (size + 3) & ~3; /* Keep memory 32-bit aligned */
 | |
| 
 | |
|     return x;
 | |
|     (void)reason;
 | |
| }
 | |
| 
 | |
| void *mpeg_malloc(size_t size, mpeg2_alloc_t reason)
 | |
| {
 | |
|     return mpeg_malloc_internal(mallocbuf, &mem_ptr, bufsize, size,
 | |
|                                 reason);
 | |
| }
 | |
| 
 | |
| void *mpeg_malloc_all(size_t *size_out, mpeg2_alloc_t reason)
 | |
| {
 | |
|     /* Can steal all but MIN_MEMMARGIN */
 | |
|     if (bufsize - mem_ptr < MIN_MEMMARGIN)
 | |
|         return NULL;
 | |
| 
 | |
|     *size_out = bufsize - mem_ptr - MIN_MEMMARGIN;
 | |
|     return mpeg_malloc(*size_out, reason);
 | |
| }
 | |
| 
 | |
| bool mpeg_alloc_init(unsigned char *buf, size_t mallocsize)
 | |
| {
 | |
|     mem_ptr = 0;
 | |
|     /* Cache-align buffer or 4-byte align */
 | |
|     mallocbuf = buf;
 | |
|     bufsize = align_buffer(PUN_PTR(void **, &mallocbuf),
 | |
|                            mallocsize, CACHEALIGN_UP(4));
 | |
| 
 | |
|     /* Separate allocator for video */
 | |
|     mpeg2_mem_ptr = 0;
 | |
|     mpeg2_mallocbuf = mallocbuf;
 | |
|     mpeg2_bufallocbuf = mallocbuf;
 | |
|     mpeg2_bufsize = CACHEALIGN_UP(LIBMPEG2_ALLOC_SIZE);
 | |
| 
 | |
|     if (mpeg_malloc_internal(mallocbuf, &mem_ptr,
 | |
|                              bufsize, mpeg2_bufsize,
 | |
|                              MPEG_ALLOC_MPEG2_BUFFER) == NULL)
 | |
|     {
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     IF_COP(flush_icache());
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| /* gcc may want to use memcpy before rb is initialised, so here's a trivial
 | |
|    implementation */
 | |
| 
 | |
| void *memcpy(void *dest, const void *src, size_t n)
 | |
| {
 | |
|     size_t i;
 | |
|     char* d=(char*)dest;
 | |
|     char* s=(char*)src;
 | |
| 
 | |
|     for (i=0;i<n;i++)
 | |
|         d[i]=s[i];
 | |
| 
 | |
|     return dest;
 | |
| }
 | |
| 
 | |
| /* allocate non-dedicated buffer space which mpeg2_mem_reset will free */
 | |
| void * mpeg2_malloc(unsigned size, mpeg2_alloc_t reason)
 | |
| {
 | |
|     void *ptr = mpeg_malloc_internal(mpeg2_mallocbuf, &mpeg2_mem_ptr,
 | |
|                                      mpeg2_bufsize, size, reason);
 | |
|     /* libmpeg2 expects zero-initialized allocations */
 | |
|     if (ptr)
 | |
|         rb->memset(ptr, 0, size);
 | |
| 
 | |
|     return ptr;
 | |
| }
 | |
| 
 | |
| /* allocate dedicated buffer - memory behind buffer pointer becomes dedicated
 | |
|    so order is important */
 | |
| void * mpeg2_bufalloc(unsigned size, mpeg2_alloc_t reason)
 | |
| {
 | |
|     void *buf = mpeg2_malloc(size, reason);
 | |
| 
 | |
|     if (buf == NULL)
 | |
|         return NULL;
 | |
| 
 | |
|     mpeg2_bufallocbuf = &mpeg2_mallocbuf[mpeg2_mem_ptr];
 | |
|     return buf;
 | |
| }
 | |
| 
 | |
| /* return unused buffer portion and size */
 | |
| void * mpeg2_get_buf(size_t *size)
 | |
| {
 | |
|     if ((size_t)mpeg2_mem_ptr + 32 >= mpeg2_bufsize)
 | |
|         return NULL;
 | |
| 
 | |
|     *size = mpeg2_bufsize - mpeg2_mem_ptr;
 | |
|     return &mpeg2_mallocbuf[mpeg2_mem_ptr];
 | |
| }
 | |
| 
 | |
| /* de-allocate all non-dedicated buffer space */
 | |
| void mpeg2_mem_reset(void)
 | |
| {
 | |
|     DEBUGF("mpeg2_mem_reset\n");
 | |
|     mpeg2_mem_ptr = mpeg2_bufallocbuf - mpeg2_mallocbuf;
 | |
| }
 | |
| 
 | |
| /* The following are expected by libmad */
 | |
| void * codec_malloc(size_t size)
 | |
| {
 | |
|     void* ptr;
 | |
| 
 | |
|     ptr = mpeg_malloc_internal(mallocbuf, &mem_ptr,
 | |
|                                 bufsize, size, MPEG_ALLOC_CODEC_MALLOC);
 | |
| 
 | |
|     if (ptr)
 | |
|         rb->memset(ptr,0,size);
 | |
| 
 | |
|     return ptr;
 | |
| }
 | |
| 
 | |
| void * codec_calloc(size_t nmemb, size_t size)
 | |
| {
 | |
|     void* ptr;
 | |
| 
 | |
|     ptr = mpeg_malloc_internal(mallocbuf, &mem_ptr,
 | |
|                                bufsize, nmemb*size,
 | |
|                                MPEG_ALLOC_CODEC_CALLOC);
 | |
| 
 | |
|     if (ptr)
 | |
|         rb->memset(ptr,0,size);
 | |
| 
 | |
|     return ptr;
 | |
| }
 | |
| 
 | |
| void codec_free(void* ptr)
 | |
| {
 | |
|     DEBUGF("codec_free - %p\n", ptr);
 | |
| #if 0
 | |
|     mem_ptr = (void *)ptr - (void *)mallocbuf;
 | |
| #endif
 | |
|     (void)ptr;
 | |
| }
 | |
| 
 | |
| void *memmove(void *dest, const void *src, size_t n)
 | |
| {
 | |
|     return rb->memmove(dest,src,n);
 | |
| }
 | |
| 
 | |
| void *memset(void *s, int c, size_t n)
 | |
| {
 | |
|     return rb->memset(s,c,n);
 | |
| }
 | |
| 
 | |
| void abort(void)
 | |
| {
 | |
|     rb->lcd_putsxy(0,0,"ABORT!");
 | |
|     rb->lcd_update();
 | |
| 
 | |
|     while (1);
 | |
|     /* Let's hope this is never called */
 | |
| }
 |