2007-06-05 07:03:30 +00:00
|
|
|
/***************************************************************************
|
|
|
|
* __________ __ ___.
|
|
|
|
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
|
|
|
|
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
|
|
|
|
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
|
|
|
|
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
|
|
|
|
* \/ \/ \/ \/ \/
|
|
|
|
* $Id$
|
|
|
|
*
|
|
|
|
* Copyright (C) 2007 by Michael Sevakis
|
|
|
|
*
|
2008-06-28 18:10:04 +00:00
|
|
|
* 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.
|
2007-06-05 07:03:30 +00:00
|
|
|
*
|
|
|
|
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
|
|
|
|
* KIND, either express or implied.
|
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
#include "system.h"
|
|
|
|
#include "cpu.h"
|
|
|
|
#include "audio.h"
|
|
|
|
#include "sound.h"
|
2010-06-26 10:07:17 +00:00
|
|
|
#include "general.h"
|
2007-06-05 07:03:30 +00:00
|
|
|
|
|
|
|
int audio_channels = 2;
|
|
|
|
int audio_output_source = AUDIO_SRC_PLAYBACK;
|
|
|
|
|
|
|
|
void audio_set_output_source(int source)
|
|
|
|
{
|
|
|
|
int oldmode = set_fiq_status(FIQ_DISABLED);
|
|
|
|
|
|
|
|
if ((unsigned)source >= AUDIO_NUM_SOURCES)
|
|
|
|
source = AUDIO_SRC_PLAYBACK;
|
|
|
|
|
|
|
|
audio_output_source = source;
|
|
|
|
|
|
|
|
if (source != AUDIO_SRC_PLAYBACK)
|
|
|
|
IISCONFIG |= (1 << 29);
|
|
|
|
|
|
|
|
set_fiq_status(oldmode);
|
|
|
|
} /* audio_set_output_source */
|
|
|
|
|
2007-06-08 23:42:04 +00:00
|
|
|
void audio_input_mux(int source, unsigned flags)
|
2007-06-05 07:03:30 +00:00
|
|
|
{
|
|
|
|
static int last_source = AUDIO_SRC_PLAYBACK;
|
2010-09-17 20:42:45 +00:00
|
|
|
#ifdef HAVE_RECORDING
|
2007-06-05 07:03:30 +00:00
|
|
|
static bool last_recording = false;
|
|
|
|
bool recording = flags & SRCF_RECORDING;
|
2010-09-17 20:42:45 +00:00
|
|
|
#else
|
|
|
|
(void) flags;
|
|
|
|
#endif
|
2007-06-05 07:03:30 +00:00
|
|
|
|
|
|
|
switch (source)
|
|
|
|
{
|
|
|
|
default: /* playback - no recording */
|
|
|
|
source = AUDIO_SRC_PLAYBACK;
|
|
|
|
case AUDIO_SRC_PLAYBACK:
|
|
|
|
audio_channels = 2;
|
|
|
|
if (source != last_source)
|
|
|
|
{
|
2010-09-17 20:42:45 +00:00
|
|
|
#if defined(HAVE_RECORDING) || defined(HAVE_FMRADIO_IN)
|
2007-06-05 07:03:30 +00:00
|
|
|
audiohw_set_monitor(false);
|
2010-09-17 20:42:45 +00:00
|
|
|
#endif
|
|
|
|
#ifdef HAVE_RECORDING
|
2007-06-05 07:03:30 +00:00
|
|
|
audiohw_disable_recording();
|
2010-09-17 20:42:45 +00:00
|
|
|
#endif
|
2007-06-05 07:03:30 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2010-09-17 20:42:45 +00:00
|
|
|
#if defined(HAVE_RECORDING) && (INPUT_SRC_CAPS & SRC_CAP_MIC)
|
2007-06-05 07:03:30 +00:00
|
|
|
case AUDIO_SRC_MIC: /* recording only */
|
|
|
|
audio_channels = 1;
|
|
|
|
if (source != last_source)
|
|
|
|
{
|
|
|
|
audiohw_set_monitor(false);
|
|
|
|
audiohw_enable_recording(true); /* source mic */
|
|
|
|
}
|
|
|
|
break;
|
2010-09-17 20:42:45 +00:00
|
|
|
#endif
|
2007-06-05 07:03:30 +00:00
|
|
|
|
2010-09-17 20:42:45 +00:00
|
|
|
#if (INPUT_SRC_CAPS & SRC_CAP_FMRADIO)
|
2007-06-05 07:03:30 +00:00
|
|
|
case AUDIO_SRC_FMRADIO: /* recording and playback */
|
|
|
|
audio_channels = 2;
|
|
|
|
|
2010-09-17 20:42:45 +00:00
|
|
|
if (source == last_source
|
|
|
|
#ifdef HAVE_RECORDING
|
|
|
|
&& recording == last_recording
|
|
|
|
#endif
|
|
|
|
)
|
2007-06-05 07:03:30 +00:00
|
|
|
break;
|
|
|
|
|
2010-09-17 20:42:45 +00:00
|
|
|
#ifdef HAVE_RECORDING
|
2007-06-05 07:03:30 +00:00
|
|
|
last_recording = recording;
|
|
|
|
if (recording)
|
|
|
|
{
|
|
|
|
audiohw_set_monitor(false);
|
|
|
|
audiohw_enable_recording(false);
|
|
|
|
}
|
2010-09-17 20:42:45 +00:00
|
|
|
#endif
|
2007-06-05 07:03:30 +00:00
|
|
|
else
|
|
|
|
{
|
2010-09-17 20:42:45 +00:00
|
|
|
#ifdef HAVE_RECORDING
|
2007-06-05 07:03:30 +00:00
|
|
|
audiohw_disable_recording();
|
2010-09-17 20:42:45 +00:00
|
|
|
#endif
|
|
|
|
#if defined(HAVE_RECORDING) || defined(HAVE_FMRADIO_IN)
|
2007-06-05 07:03:30 +00:00
|
|
|
audiohw_set_monitor(true); /* line 1 analog audio path */
|
2010-09-17 20:42:45 +00:00
|
|
|
#endif
|
|
|
|
|
2007-06-05 07:03:30 +00:00
|
|
|
}
|
|
|
|
break;
|
2010-09-17 20:42:45 +00:00
|
|
|
#endif /* (INPUT_SRC_CAPS & SRC_CAP_FMRADIO) */
|
2007-06-05 07:03:30 +00:00
|
|
|
} /* end switch */
|
|
|
|
|
|
|
|
last_source = source;
|
2007-06-08 23:42:04 +00:00
|
|
|
} /* audio_input_mux */
|
2010-06-26 10:07:17 +00:00
|
|
|
|
|
|
|
|
|
|
|
void audiohw_set_sampr_dividers(int fsel)
|
|
|
|
{
|
|
|
|
/* Seems to predivide 24MHz by 2 for a source clock of 12MHz. Maybe
|
|
|
|
* there's a way to set that? */
|
|
|
|
static const struct
|
|
|
|
{
|
|
|
|
unsigned char iisclk;
|
|
|
|
unsigned char iisdiv;
|
|
|
|
} regvals[HW_NUM_FREQ] =
|
|
|
|
{
|
|
|
|
/* 8kHz - 24kHz work well but there seems to be a minor crackling
|
|
|
|
* issue for playback at times and all possibilities were checked
|
|
|
|
* for the divisors without any positive change.
|
|
|
|
* 32kHz - 48kHz seem fine all around. */
|
|
|
|
#if 0
|
|
|
|
[HW_FREQ_8] = /* CLK / 1500 (8000Hz) */
|
|
|
|
{
|
|
|
|
.iisclk = 24,
|
|
|
|
.iisdiv = 59,
|
|
|
|
},
|
|
|
|
[HW_FREQ_11] = /* CLK / 1088 (~11029.41Hz) */
|
|
|
|
{
|
|
|
|
.iisclk = 33,
|
|
|
|
.iisdiv = 31,
|
|
|
|
},
|
|
|
|
[HW_FREQ_12] = /* CLK / 1000 (120000Hz) */
|
|
|
|
{
|
|
|
|
.iisclk = 49,
|
|
|
|
.iisdiv = 19,
|
|
|
|
},
|
|
|
|
[HW_FREQ_16] = /* CLK / 750 (16000Hz) */
|
|
|
|
{
|
|
|
|
.iisclk = 24,
|
|
|
|
.iisdiv = 29,
|
|
|
|
},
|
|
|
|
[HW_FREQ_22] = /* CLK / 544 (~22058.82Hz) */
|
|
|
|
{
|
|
|
|
.iisclk = 33,
|
|
|
|
.iisdiv = 15,
|
|
|
|
},
|
|
|
|
[HW_FREQ_24] = /* CLK / 500 (24000Hz) */
|
|
|
|
{
|
|
|
|
.iisclk = 49,
|
|
|
|
.iisdiv = 9,
|
|
|
|
},
|
|
|
|
#endif
|
|
|
|
[HW_FREQ_32] = /* CLK / 375 (32000Hz) */
|
|
|
|
{
|
|
|
|
.iisclk = 24,
|
|
|
|
.iisdiv = 14,
|
|
|
|
},
|
|
|
|
[HW_FREQ_44] = /* CLK / 272 (~44117.68Hz) */
|
|
|
|
{
|
|
|
|
.iisclk = 33,
|
|
|
|
.iisdiv = 7,
|
|
|
|
},
|
|
|
|
[HW_FREQ_48] = /* CLK / 250 (48000Hz) */
|
|
|
|
{
|
|
|
|
.iisclk = 49,
|
|
|
|
.iisdiv = 4,
|
|
|
|
},
|
|
|
|
/* going a bit higher would be nice to get 64kHz play, 32kHz rec, but a
|
|
|
|
* close enough division isn't obtainable unless CLK can be changed */
|
|
|
|
};
|
|
|
|
|
|
|
|
IISCLK = (IISCLK & ~0x17ff) | regvals[fsel].iisclk;
|
|
|
|
IISDIV = (IISDIV & ~0xc000003f) | regvals[fsel].iisdiv;
|
|
|
|
}
|
|
|
|
|
2011-12-08 19:20:00 +00:00
|
|
|
#ifdef CONFIG_SAMPR_TYPES
|
|
|
|
unsigned int pcm_sampr_to_hw_sampr(unsigned int samplerate,
|
|
|
|
unsigned int type)
|
2010-06-26 10:07:17 +00:00
|
|
|
{
|
2011-12-08 19:20:00 +00:00
|
|
|
#ifdef HAVE_RECORDING
|
|
|
|
if (samplerate != HW_SAMPR_RESET && type == SAMPR_TYPE_REC)
|
|
|
|
{
|
|
|
|
/* Check if the samplerate is in the list of recordable rates.
|
|
|
|
* Fail to default if not */
|
|
|
|
int index = round_value_to_list32(samplerate, rec_freq_sampr,
|
|
|
|
REC_NUM_FREQ, false);
|
|
|
|
if (samplerate != rec_freq_sampr[index])
|
|
|
|
samplerate = REC_SAMPR_DEFAULT;
|
|
|
|
|
|
|
|
samplerate *= 2; /* Recording rates are 1/2 the codec clock */
|
|
|
|
}
|
|
|
|
#endif /* HAVE_RECORDING */
|
|
|
|
|
|
|
|
return samplerate;
|
|
|
|
(void)type;
|
2010-06-26 10:07:17 +00:00
|
|
|
}
|
2011-12-08 19:20:00 +00:00
|
|
|
#endif /* CONFIG_SAMPR_TYPES */
|