forked from len0rd/rockbox
		
	system.h doesn't need it on its own and this change makes it less dependant on Rockbox internals. Change-Id: I4e1e4108a52a7b599627a829204eb82b392fc6d6
		
			
				
	
	
		
			752 lines
		
	
	
	
		
			20 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			752 lines
		
	
	
	
		
			20 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /***************************************************************************
 | |
|  *             __________               __   ___.
 | |
|  *   Open      \______   \ ____   ____ |  | _\_ |__   _______  ___
 | |
|  *   Source     |       _//  _ \_/ ___\|  |/ /| __ \ /  _ \  \/  /
 | |
|  *   Jukebox    |    |   (  <_> )  \___|    < | \_\ (  <_> > <  <
 | |
|  *   Firmware   |____|_  /\____/ \___  >__|_ \|___  /\____/__/\_ \
 | |
|  *                     \/            \/     \/    \/            \/
 | |
|  * $Id$
 | |
|  *
 | |
|  * Copyright (C) 2005-2007 Miika Pekkarinen
 | |
|  * Copyright (C) 2007-2008 Nicolas Pennequin
 | |
|  * Copyright (C) 2011      Michael Sevakis
 | |
|  *
 | |
|  * 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 software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
 | |
|  * KIND, either express or implied.
 | |
|  *
 | |
|  ****************************************************************************/
 | |
| 
 | |
| #include "config.h"
 | |
| #include "system.h"
 | |
| #include "kernel.h"
 | |
| #include "codecs.h"
 | |
| #include "codec_thread.h"
 | |
| #include "pcmbuf.h"
 | |
| #include "audio_thread.h"
 | |
| #include "playback.h"
 | |
| #include "buffering.h"
 | |
| #include "dsp_core.h"
 | |
| #include "metadata.h"
 | |
| #include "settings.h"
 | |
| 
 | |
| /* Define LOGF_ENABLE to enable logf output in this file */
 | |
| /*#define LOGF_ENABLE*/
 | |
| #include "logf.h"
 | |
| 
 | |
| /* macros to enable logf for queues
 | |
|    logging on SYS_TIMEOUT can be disabled */
 | |
| #ifdef SIMULATOR
 | |
| /* Define this for logf output of all queuing except SYS_TIMEOUT */
 | |
| #define PLAYBACK_LOGQUEUES
 | |
| /* Define this to logf SYS_TIMEOUT messages */
 | |
| /*#define PLAYBACK_LOGQUEUES_SYS_TIMEOUT*/
 | |
| #endif
 | |
| 
 | |
| #ifdef PLAYBACK_LOGQUEUES
 | |
| #define LOGFQUEUE logf
 | |
| #else
 | |
| #define LOGFQUEUE(...)
 | |
| #endif
 | |
| 
 | |
| #ifdef PLAYBACK_LOGQUEUES_SYS_TIMEOUT
 | |
| #define LOGFQUEUE_SYS_TIMEOUT logf
 | |
| #else
 | |
| #define LOGFQUEUE_SYS_TIMEOUT(...)
 | |
| #endif
 | |
| 
 | |
| /* Variables are commented with the threads that use them:
 | |
|  * A=audio, C=codec
 | |
|  * - = reads only
 | |
|  *
 | |
|  * Unless otherwise noted, the extern variables are located
 | |
|  * in playback.c.
 | |
|  */
 | |
| 
 | |
| /* Q_LOAD_CODEC parameter data */
 | |
| struct codec_load_info
 | |
| {
 | |
|     int hid;    /* audio handle id (specify < 0 to use afmt) */
 | |
|     int afmt;   /* codec specification (AFMT_*) */
 | |
| };
 | |
| 
 | |
| 
 | |
| /** --- Main state control --- **/
 | |
| 
 | |
| static int codec_type = AFMT_UNKNOWN; /* Codec type (C,A-) */
 | |
| 
 | |
| /* Private interfaces to main playback control */
 | |
| extern void audio_codec_update_elapsed(unsigned long elapsed);
 | |
| extern void audio_codec_update_offset(size_t offset);
 | |
| extern void audio_codec_complete(int status);
 | |
| extern void audio_codec_seek_complete(void);
 | |
| extern struct codec_api ci; /* from codecs.c */
 | |
| 
 | |
| /* Codec thread */
 | |
| static unsigned int codec_thread_id; /* For modifying thread priority later */
 | |
| static struct event_queue codec_queue SHAREDBSS_ATTR;
 | |
| static struct queue_sender_list codec_queue_sender_list SHAREDBSS_ATTR;
 | |
| static long codec_stack[(DEFAULT_STACK_SIZE + 0x2000)/sizeof(long)] IBSS_ATTR;
 | |
| static const char codec_thread_name[] = "codec";
 | |
| 
 | |
| static void unload_codec(void);
 | |
| 
 | |
| /* Messages are only ever sent one at a time to the codec from the audio
 | |
|    thread. This is important for correct operation unless playback is
 | |
|    stopped. */
 | |
| 
 | |
| /* static routines */
 | |
| static void codec_queue_ack(intptr_t ackme)
 | |
| {
 | |
|     queue_reply(&codec_queue, ackme);
 | |
| }
 | |
| 
 | |
| static intptr_t codec_queue_send(long id, intptr_t data)
 | |
| {
 | |
|     return queue_send(&codec_queue, id, data);
 | |
| }
 | |
| 
 | |
| /* Poll the state of the codec queue. Returns < 0 if the message is urgent
 | |
|    and any state should exit, > 0 if it's a run message (and it was
 | |
|    scrubbed), 0 if message was ignored. */
 | |
| static int codec_check_queue__have_msg(void)
 | |
| {
 | |
|     struct queue_event ev;
 | |
| 
 | |
|     queue_peek(&codec_queue, &ev);
 | |
| 
 | |
|     /* Seek, pause or stop? Just peek and return if so. Codec
 | |
|        must handle the command after returing. Inserts will not
 | |
|        be allowed until it complies. */
 | |
|     switch (ev.id)
 | |
|     {
 | |
|     case Q_CODEC_SEEK:
 | |
|         LOGFQUEUE("codec - Q_CODEC_SEEK", ev.id);
 | |
|         return -1;
 | |
|     case Q_CODEC_PAUSE:
 | |
|         LOGFQUEUE("codec - Q_CODEC_PAUSE", ev.id);
 | |
|         return -1;
 | |
|     case Q_CODEC_STOP:
 | |
|         LOGFQUEUE("codec - Q_CODEC_STOP", ev.id);
 | |
|         return -1;
 | |
|     }
 | |
| 
 | |
|     /* This is in error in this context unless it's "go, go, go!" */
 | |
|     queue_wait(&codec_queue, &ev);
 | |
| 
 | |
|     if (ev.id == Q_CODEC_RUN)
 | |
|     {
 | |
|         logf("codec < Q_CODEC_RUN: already running!");
 | |
|         codec_queue_ack(Q_CODEC_RUN);
 | |
|         return 1;
 | |
|     }
 | |
| 
 | |
|     /* Ignore it */
 | |
|     logf("codec < bad req %ld (%s)", ev.id, __func__);
 | |
|     codec_queue_ack(Q_NULL);
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| /* Does the audio format type equal CODEC_TYPE_ENCODER? */
 | |
| static inline bool type_is_encoder(int afmt)
 | |
| {
 | |
| #ifdef AUDIO_HAVE_RECORDING
 | |
|     return (afmt & CODEC_TYPE_MASK) == CODEC_TYPE_ENCODER;
 | |
| #else
 | |
|     return false;
 | |
|     (void)afmt;
 | |
| #endif
 | |
| }
 | |
| 
 | |
| /**************************************/
 | |
| 
 | |
| 
 | |
| /** --- Miscellaneous external functions --- **/
 | |
| const char * get_codec_filename(int cod_spec)
 | |
| {
 | |
|     const char *fname;
 | |
| 
 | |
| #ifdef HAVE_RECORDING
 | |
|     /* Can choose decoder or encoder if one available */
 | |
|     int type = cod_spec & CODEC_TYPE_MASK;
 | |
|     int afmt = cod_spec & CODEC_AFMT_MASK;
 | |
| 
 | |
|     if ((unsigned)afmt >= AFMT_NUM_CODECS)
 | |
|         type = AFMT_UNKNOWN | (type & CODEC_TYPE_MASK);
 | |
| 
 | |
|     fname = (type == CODEC_TYPE_ENCODER) ?
 | |
|                 audio_formats[afmt].codec_enc_root_fn :
 | |
|                 audio_formats[afmt].codec_root_fn;
 | |
| 
 | |
|     logf("%s: %d - %s",
 | |
|         (type == CODEC_TYPE_ENCODER) ? "Encoder" : "Decoder",
 | |
|         afmt, fname ? fname : "<unknown>");
 | |
| #else /* !HAVE_RECORDING */
 | |
|     /* Always decoder */
 | |
|     if ((unsigned)cod_spec >= AFMT_NUM_CODECS)
 | |
|         cod_spec = AFMT_UNKNOWN;
 | |
|     fname = audio_formats[cod_spec].codec_root_fn;
 | |
|     logf("Codec: %d - %s",  cod_spec, fname ? fname : "<unknown>");
 | |
| #endif /* HAVE_RECORDING */
 | |
| 
 | |
|     return fname;
 | |
| }
 | |
| 
 | |
| /* Borrow the codec thread and return the ID */
 | |
| void codec_thread_do_callback(void (*fn)(void), unsigned int *id)
 | |
| {
 | |
|     /* Set id before telling thread to call something; it may be
 | |
|      * needed before this function returns. */
 | |
|     if (id != NULL)
 | |
|         *id = codec_thread_id;
 | |
| 
 | |
|     /* Codec thread will signal just before entering callback */
 | |
|     LOGFQUEUE("codec >| Q_CODEC_DO_CALLBACK");
 | |
|     codec_queue_send(Q_CODEC_DO_CALLBACK, (intptr_t)fn);
 | |
| }
 | |
| 
 | |
| 
 | |
| /** --- codec API callbacks --- **/
 | |
| 
 | |
| static void codec_pcmbuf_insert_callback(
 | |
|         const void *ch1, const void *ch2, int count)
 | |
| {
 | |
|     struct dsp_buffer src;
 | |
|     src.remcount  = count;
 | |
|     src.pin[0]    = ch1;
 | |
|     src.pin[1]    = ch2;
 | |
|     src.proc_mask = 0;
 | |
| 
 | |
|     while (LIKELY(queue_empty(&codec_queue)) ||
 | |
|            codec_check_queue__have_msg() >= 0)
 | |
|     {
 | |
|         struct dsp_buffer dst;
 | |
|         dst.remcount = 0;
 | |
|         dst.bufcount = MAX(src.remcount, 1024); /* Arbitrary min request */
 | |
| 
 | |
|         if ((dst.p16out = pcmbuf_request_buffer(&dst.bufcount)) == NULL)
 | |
|         {
 | |
|             cancel_cpu_boost();
 | |
| 
 | |
|             /* It may be awhile before space is available but we want
 | |
|                "instant" response to any message */
 | |
|             queue_wait_w_tmo(&codec_queue, NULL, HZ/20);
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             dsp_process(ci.dsp, &src, &dst);
 | |
| 
 | |
|             if (dst.remcount > 0)
 | |
|             {
 | |
|                 pcmbuf_write_complete(dst.remcount, ci.id3->elapsed,
 | |
|                                       ci.id3->offset);
 | |
|             }
 | |
|             else if (src.remcount <= 0)
 | |
|             {
 | |
|                 return; /* No input remains and DSP purged */
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| /* helper function, not a callback */
 | |
| static bool codec_advance_buffer_counters(size_t amount)
 | |
| {
 | |
|     if (bufadvance(ci.audio_hid, amount) < 0)
 | |
|     {
 | |
|         ci.curpos = ci.filesize;
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     ci.curpos += amount;
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| /* copy up-to size bytes into ptr and return the actual size copied */
 | |
| static size_t codec_filebuf_callback(void *ptr, size_t size)
 | |
| {
 | |
|     ssize_t copy_n = bufread(ci.audio_hid, size, ptr);
 | |
| 
 | |
|     /* Nothing requested OR nothing left */
 | |
|     if (copy_n <= 0)
 | |
|         return 0;
 | |
| 
 | |
|     /* Update read and other position pointers */
 | |
|     codec_advance_buffer_counters(copy_n);
 | |
| 
 | |
|     /* Return the actual amount of data copied to the buffer */
 | |
|     return copy_n;
 | |
| }
 | |
| 
 | |
| static void * codec_request_buffer_callback(size_t *realsize, size_t reqsize)
 | |
| {
 | |
|     size_t copy_n = reqsize;
 | |
|     ssize_t ret;
 | |
|     void *ptr;
 | |
| 
 | |
|     ret = bufgetdata(ci.audio_hid, reqsize, &ptr);
 | |
|     if (ret >= 0)
 | |
|         copy_n = MIN((size_t)ret, reqsize);
 | |
|     else
 | |
|         copy_n = 0;
 | |
| 
 | |
|     if (copy_n == 0)
 | |
|         ptr = NULL;
 | |
| 
 | |
|     *realsize = copy_n;
 | |
|     return ptr;
 | |
| }
 | |
| 
 | |
| static void codec_advance_buffer_callback(size_t amount)
 | |
| {
 | |
|     if (!codec_advance_buffer_counters(amount))
 | |
|         return;
 | |
| 
 | |
|     audio_codec_update_offset(ci.curpos);
 | |
| }
 | |
| 
 | |
| static bool codec_seek_buffer_callback(size_t newpos)
 | |
| {
 | |
|     logf("codec_seek_buffer_callback");
 | |
| 
 | |
|     int ret = bufseek(ci.audio_hid, newpos);
 | |
|     if (ret == 0)
 | |
|     {
 | |
|         ci.curpos = newpos;
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| static void codec_seek_complete_callback(void)
 | |
| {
 | |
|     logf("seek_complete");
 | |
| 
 | |
|     /* Clear DSP */
 | |
|     dsp_configure(ci.dsp, DSP_FLUSH, 0);
 | |
| 
 | |
|     /* Sync position */
 | |
|     audio_codec_update_offset(ci.curpos);
 | |
| 
 | |
|     /* Post notification to audio thread */
 | |
|     audio_codec_seek_complete();
 | |
| 
 | |
|     /* Wait for urgent or go message */
 | |
|     do
 | |
|     {
 | |
|         queue_wait(&codec_queue, NULL);
 | |
|     }
 | |
|     while (codec_check_queue__have_msg() == 0);
 | |
| }
 | |
| 
 | |
| static void codec_configure_callback(int setting, intptr_t value)
 | |
| {
 | |
|     dsp_configure(ci.dsp, setting, value);
 | |
| }
 | |
| 
 | |
| static enum codec_command_action
 | |
|     codec_get_command_callback(intptr_t *param)
 | |
| {
 | |
|     yield();
 | |
| 
 | |
|     if (LIKELY(queue_empty(&codec_queue)))
 | |
|         return CODEC_ACTION_NULL; /* As you were */
 | |
| 
 | |
|     /* Process the message - return requested action and data (if any should
 | |
|        be expected) */
 | |
|     while (1)
 | |
|     {
 | |
|         enum codec_command_action action = CODEC_ACTION_NULL;
 | |
|         struct queue_event ev;
 | |
| 
 | |
|         queue_peek(&codec_queue, &ev); /* Find out what it is */
 | |
| 
 | |
|         intptr_t id = ev.id;
 | |
| 
 | |
|         switch (id)
 | |
|         {
 | |
|         case Q_NULL:
 | |
|             LOGFQUEUE("codec < Q_NULL");
 | |
|             break;
 | |
| 
 | |
|         case Q_CODEC_RUN:   /* Already running */
 | |
|             LOGFQUEUE("codec < Q_CODEC_RUN");
 | |
|             break;
 | |
| 
 | |
|         case Q_CODEC_PAUSE: /* Stay here and wait */
 | |
|             LOGFQUEUE("codec < Q_CODEC_PAUSE");
 | |
|             queue_wait(&codec_queue, &ev);  /* Remove message */
 | |
|             codec_queue_ack(Q_CODEC_PAUSE);
 | |
|             queue_wait(&codec_queue, NULL); /* Wait for next (no remove) */
 | |
|             continue;
 | |
| 
 | |
|         case Q_CODEC_SEEK:  /* Audio wants codec to seek */
 | |
|             LOGFQUEUE("codec < Q_CODEC_SEEK %ld", ev.data);
 | |
|             *param = ev.data;
 | |
|             action = CODEC_ACTION_SEEK_TIME;
 | |
|             trigger_cpu_boost();
 | |
|             break;
 | |
| 
 | |
|         case Q_CODEC_STOP:  /* Must only return 0 in main loop */
 | |
|             LOGFQUEUE("codec < Q_CODEC_STOP: %ld", ev.data);
 | |
| #ifdef HAVE_RECORDING
 | |
|             if (type_is_encoder(codec_type))
 | |
|             {
 | |
|                 /* Stream finish request (soft stop)? */
 | |
|                 if (ev.data && param)
 | |
|                 {
 | |
|                     /* ev.data is pointer to size */
 | |
|                     *param = ev.data;
 | |
|                     action = CODEC_ACTION_STREAM_FINISH;
 | |
|                     break;
 | |
|                 }
 | |
|             }
 | |
|             else
 | |
| #endif /* HAVE_RECORDING */
 | |
|             {
 | |
|                 dsp_configure(ci.dsp, DSP_FLUSH, 0); /* Discontinuity */
 | |
|             }
 | |
| 
 | |
|             return CODEC_ACTION_HALT; /* Leave in queue */
 | |
| 
 | |
|         default:            /* This is in error in this context. */
 | |
|             logf("codec bad req %ld (%s)", ev.id, __func__);
 | |
|             id = Q_NULL;
 | |
|         }
 | |
| 
 | |
|         queue_wait(&codec_queue, &ev); /* Actually remove it */
 | |
|         codec_queue_ack(id);
 | |
|         return action;
 | |
|     }
 | |
| }
 | |
| 
 | |
| static bool codec_loop_track_callback(void)
 | |
| {
 | |
|     return global_settings.repeat_mode == REPEAT_ONE;
 | |
| }
 | |
| 
 | |
| 
 | |
| /** --- CODEC THREAD --- **/
 | |
| 
 | |
| /* Handle Q_CODEC_LOAD */
 | |
| static void load_codec(const struct codec_load_info *ev_data)
 | |
| {
 | |
|     int status = CODEC_ERROR;
 | |
|     /* Save a local copy so we can let the audio thread go ASAP */
 | |
|     struct codec_load_info data = *ev_data;
 | |
|     bool const encoder = type_is_encoder(data.afmt);
 | |
| 
 | |
|     if (codec_type != AFMT_UNKNOWN)
 | |
|     {
 | |
|         /* Must have unloaded it first */
 | |
|         logf("a codec is already loaded");
 | |
|         if (data.hid >= 0)
 | |
|             bufclose(data.hid);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     trigger_cpu_boost();
 | |
| 
 | |
|     if (!encoder)
 | |
|     {
 | |
|         /* Do this now because codec may set some things up at load time */
 | |
|         dsp_configure(ci.dsp, DSP_RESET, 0);
 | |
|     }
 | |
| 
 | |
|     if (data.hid >= 0)
 | |
|     {
 | |
|         /* First try buffer load */
 | |
|         status = codec_load_buf(data.hid, &ci);
 | |
|         bufclose(data.hid);
 | |
|     }
 | |
| 
 | |
|     if (status < 0)
 | |
|     {
 | |
|         /* Either not a valid handle or the buffer method failed */
 | |
|         const char *codec_fn = get_codec_filename(data.afmt);
 | |
|         if (codec_fn)
 | |
|         {
 | |
| #ifdef HAVE_IO_PRIORITY
 | |
|             buf_back_off_storage(true);
 | |
| #endif
 | |
|             status = codec_load_file(codec_fn, &ci);
 | |
| #ifdef HAVE_IO_PRIORITY
 | |
|             buf_back_off_storage(false);
 | |
| #endif
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /* Types must agree */
 | |
|     if (status >= 0 && encoder == !!codec_get_enc_callback())
 | |
|     {
 | |
|         codec_type = data.afmt;
 | |
|         codec_queue_ack(Q_CODEC_LOAD);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Failed - get rid of it */
 | |
|     unload_codec();
 | |
| }
 | |
| 
 | |
| /* Handle Q_CODEC_RUN */
 | |
| static void run_codec(void)
 | |
| {
 | |
|     bool const encoder = type_is_encoder(codec_type);
 | |
|     int status;
 | |
| 
 | |
|     if (codec_type == AFMT_UNKNOWN)
 | |
|     {
 | |
|         logf("no codec to run");
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     codec_queue_ack(Q_CODEC_RUN);
 | |
| 
 | |
|     trigger_cpu_boost();
 | |
|     dsp_configure(ci.dsp, DSP_SET_OUT_FREQUENCY, pcmbuf_get_frequency());
 | |
| 
 | |
|     if (!encoder)
 | |
|     {
 | |
|         /* This will be either the initial buffered offset or where it left off
 | |
|            if it remained buffered and we're skipping back to it and it is best
 | |
|            to have ci.curpos in sync with the handle's read position - it's the
 | |
|            codec's responsibility to ensure it has the correct positions -
 | |
|            playback is sorta dumb and only has a vague idea about what to
 | |
|            buffer based upon what metadata has to say */
 | |
|         ci.curpos = bufftell(ci.audio_hid);
 | |
| 
 | |
|         /* Pin the codec's audio data in place */
 | |
|         buf_pin_handle(ci.audio_hid, true);
 | |
|     }
 | |
| 
 | |
|     status = codec_run_proc();
 | |
| 
 | |
|     if (!encoder)
 | |
|     {
 | |
|         /* Codec is done with it - let it move */
 | |
|         buf_pin_handle(ci.audio_hid, false);
 | |
| 
 | |
|         /* Notify audio that we're done for better or worse - advise of the
 | |
|            status */
 | |
|         audio_codec_complete(status);
 | |
|     }
 | |
| }
 | |
| 
 | |
| /* Handle Q_CODEC_SEEK */
 | |
| static void seek_codec(unsigned long time)
 | |
| {
 | |
|     if (codec_type == AFMT_UNKNOWN)
 | |
|     {
 | |
|         logf("no codec to seek");
 | |
|         codec_queue_ack(Q_CODEC_SEEK);
 | |
|         codec_seek_complete_callback();
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Post it up one level */
 | |
|     queue_post(&codec_queue, Q_CODEC_SEEK, time);
 | |
|     codec_queue_ack(Q_CODEC_SEEK);
 | |
| 
 | |
|     /* Have to run it again */
 | |
|     run_codec();
 | |
| }
 | |
| 
 | |
| /* Handle Q_CODEC_UNLOAD */
 | |
| static void unload_codec(void)
 | |
| {
 | |
|     /* Tell codec to clean up */
 | |
|     codec_type = AFMT_UNKNOWN;
 | |
|     codec_close();
 | |
| }
 | |
| 
 | |
| /* Handle Q_CODEC_DO_CALLBACK */
 | |
| static void do_callback(void (* callback)(void))
 | |
| {
 | |
|     codec_queue_ack(Q_CODEC_DO_CALLBACK);
 | |
| 
 | |
|     if (callback)
 | |
|     {
 | |
|         commit_discard_idcache();
 | |
|         callback();
 | |
|         commit_dcache();
 | |
|     }
 | |
| }
 | |
| 
 | |
| /* Codec thread function */
 | |
| static void NORETURN_ATTR codec_thread(void)
 | |
| {
 | |
|     struct queue_event ev;
 | |
| 
 | |
|     while (1)
 | |
|     {
 | |
|         cancel_cpu_boost();
 | |
| 
 | |
|         queue_wait(&codec_queue, &ev);
 | |
| 
 | |
|         switch (ev.id)
 | |
|         {
 | |
|         case Q_CODEC_LOAD:
 | |
|             LOGFQUEUE("codec < Q_CODEC_LOAD");
 | |
|             load_codec((const struct codec_load_info *)ev.data);
 | |
|             break;
 | |
| 
 | |
|         case Q_CODEC_RUN:
 | |
|             LOGFQUEUE("codec < Q_CODEC_RUN");
 | |
|             run_codec();
 | |
|             break;
 | |
| 
 | |
|         case Q_CODEC_PAUSE:
 | |
|             LOGFQUEUE("codec < Q_CODEC_PAUSE");
 | |
|             break;
 | |
| 
 | |
|         case Q_CODEC_SEEK:
 | |
|             LOGFQUEUE("codec < Q_CODEC_SEEK: %lu", (unsigned long)ev.data);
 | |
|             seek_codec(ev.data);
 | |
|             break;
 | |
| 
 | |
|         case Q_CODEC_UNLOAD:
 | |
|             LOGFQUEUE("codec < Q_CODEC_UNLOAD");
 | |
|             unload_codec();
 | |
|             break;
 | |
| 
 | |
|         case Q_CODEC_DO_CALLBACK:
 | |
|             LOGFQUEUE("codec < Q_CODEC_DO_CALLBACK");
 | |
|             do_callback((void (*)(void))ev.data);
 | |
|             break;
 | |
| 
 | |
|         default:
 | |
|             LOGFQUEUE("codec < default : %ld", ev.id);
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| 
 | |
| /** --- Miscellaneous external interfaces -- **/
 | |
| 
 | |
| /* Initialize playback's codec interface */
 | |
| void INIT_ATTR codec_thread_init(void)
 | |
| {
 | |
|     /* Init API */
 | |
|     ci.dsp              = dsp_get_config(CODEC_IDX_AUDIO);
 | |
|     ci.codec_get_buffer = codec_get_buffer_callback;
 | |
|     ci.pcmbuf_insert    = codec_pcmbuf_insert_callback;
 | |
|     ci.set_elapsed      = audio_codec_update_elapsed;
 | |
|     ci.read_filebuf     = codec_filebuf_callback;
 | |
|     ci.request_buffer   = codec_request_buffer_callback;
 | |
|     ci.advance_buffer   = codec_advance_buffer_callback;
 | |
|     ci.seek_buffer      = codec_seek_buffer_callback;
 | |
|     ci.seek_complete    = codec_seek_complete_callback;
 | |
|     ci.set_offset       = audio_codec_update_offset;
 | |
|     ci.configure        = codec_configure_callback;
 | |
|     ci.get_command      = codec_get_command_callback;
 | |
|     ci.loop_track       = codec_loop_track_callback;
 | |
| 
 | |
|     /* Init threading */
 | |
|     queue_init(&codec_queue, false);
 | |
|     codec_thread_id = create_thread(
 | |
|             codec_thread, codec_stack, sizeof(codec_stack), 0,
 | |
|             codec_thread_name IF_PRIO(, PRIORITY_PLAYBACK)
 | |
|             IF_COP(, CPU));
 | |
|     queue_enable_queue_send(&codec_queue, &codec_queue_sender_list,
 | |
|                             codec_thread_id);
 | |
| }
 | |
| 
 | |
| #ifdef HAVE_PRIORITY_SCHEDULING
 | |
| /* Obtain codec thread's current priority */
 | |
| int codec_thread_get_priority(void)
 | |
| {
 | |
|     return thread_get_priority(codec_thread_id);
 | |
| }
 | |
| 
 | |
| /* Set the codec thread's priority and return the old value */
 | |
| int codec_thread_set_priority(int priority)
 | |
| {
 | |
|     return thread_set_priority(codec_thread_id, priority);
 | |
| }
 | |
| #endif /* HAVE_PRIORITY_SCHEDULING */
 | |
| 
 | |
| 
 | |
| /** --- Functions for audio thread use --- **/
 | |
| 
 | |
| /* Load a decoder or encoder and set the format type */
 | |
| bool codec_load(int hid, int cod_spec)
 | |
| {
 | |
|     struct codec_load_info parm = { hid, cod_spec };
 | |
| 
 | |
|     LOGFQUEUE("audio >| codec Q_CODEC_LOAD: %d, %d", hid, cod_spec);
 | |
|     return codec_queue_send(Q_CODEC_LOAD, (intptr_t)&parm) != 0;
 | |
| }
 | |
| 
 | |
| /* Begin decoding the current file */
 | |
| void codec_go(void)
 | |
| {
 | |
|     LOGFQUEUE("audio >| codec Q_CODEC_RUN");
 | |
|     codec_queue_send(Q_CODEC_RUN, 0);
 | |
| }
 | |
| 
 | |
| /* Instruct the codec to seek to the specified time (should be properly
 | |
|    paused or stopped first to avoid possible buffering deadlock) */
 | |
| void codec_seek(long time)
 | |
| {
 | |
|     LOGFQUEUE("audio > codec Q_CODEC_SEEK: %ld", time);
 | |
|     codec_queue_send(Q_CODEC_SEEK, time);
 | |
| }
 | |
| 
 | |
| /* Pause the codec and make it wait for further instructions inside the
 | |
|    command callback */
 | |
| bool codec_pause(void)
 | |
| {
 | |
|     LOGFQUEUE("audio >| codec Q_CODEC_PAUSE");
 | |
|     return codec_queue_send(Q_CODEC_PAUSE, 0) != Q_NULL;
 | |
| }
 | |
| 
 | |
| /* Stop codec if running - codec stays resident if loaded */
 | |
| void codec_stop(void)
 | |
| {
 | |
|     /* Wait until it's in the main loop */
 | |
|     LOGFQUEUE("audio >| codec Q_CODEC_STOP: 0");
 | |
|     while (codec_queue_send(Q_CODEC_STOP, 0) != Q_NULL);
 | |
| }
 | |
| 
 | |
| #ifdef HAVE_RECORDING
 | |
| /* Tells codec to take final encoding step and then exit -
 | |
|    Returns minimum buffer size required or 0 if complete */
 | |
| size_t codec_finish_stream(void)
 | |
| {
 | |
|     size_t size = 0;
 | |
| 
 | |
|     LOGFQUEUE("audio >| codec Q_CODEC_STOP: &size");
 | |
|     if (codec_queue_send(Q_CODEC_STOP, (intptr_t)&size) != Q_NULL)
 | |
|     {
 | |
|         /* Sync to keep size in scope and get response */
 | |
|         LOGFQUEUE("audio >| codec Q_NULL");
 | |
|         codec_queue_send(Q_NULL, 0);
 | |
| 
 | |
|         if (size == 0)
 | |
|             codec_stop(); /* Replied with 0 size */
 | |
|     }
 | |
|     /* else thread running in the main loop */
 | |
| 
 | |
|     return size;
 | |
| }
 | |
| #endif /* HAVE_RECORDING */
 | |
| 
 | |
| /* Call the codec's exit routine and close all references */
 | |
| void codec_unload(void)
 | |
| {
 | |
|     codec_stop();
 | |
|     LOGFQUEUE("audio >| codec Q_CODEC_UNLOAD");
 | |
|     codec_queue_send(Q_CODEC_UNLOAD, 0);
 | |
| }
 | |
| 
 | |
| /* Return the afmt type of the loaded codec - sticks until calling
 | |
|    codec_unload unless initial load failed */
 | |
| int codec_loaded(void)
 | |
| {
 | |
|     return codec_type;
 | |
| }
 |