Split CPU-optimized beep code into the firmware/asm tree.
For now due to current lack of an apps/asm, place the ASM/generic code in the firmware/asm directory. Additionally, make generic beep code more generic. Change-Id: I4a69b6ffcbb97d9e6dfde2209c5a118de19e5638
This commit is contained in:
parent
a794aaa38c
commit
9a32a7b404
4 changed files with 151 additions and 61 deletions
80
apps/beep.c
80
apps/beep.c
|
@ -26,71 +26,22 @@
|
|||
#include "pcm_mixer.h"
|
||||
#include "misc.h"
|
||||
|
||||
static int32_t beep_phase; /* Phase of square wave generator */
|
||||
/** Beep generation, CPU optimized **/
|
||||
#include "../firmware/asm/beep.c"
|
||||
|
||||
static uint32_t beep_phase; /* Phase of square wave generator */
|
||||
static uint32_t beep_step; /* Step of square wave generator on each sample */
|
||||
#ifdef BEEP_GENERIC
|
||||
static int16_t beep_amplitude; /* Amplitude of square wave generator */
|
||||
#else
|
||||
/* Optimized routines do XOR with phase sign bit in both channels at once */
|
||||
static uint32_t beep_amplitude; /* Amplitude of square wave generator */
|
||||
#endif
|
||||
static int beep_count; /* Number of samples remaining to generate */
|
||||
|
||||
/* Reserve enough static space for keyclick to fit */
|
||||
#define BEEP_BUF_COUNT (NATIVE_FREQUENCY / 1000 * KEYCLICK_DURATION)
|
||||
static uint32_t beep_buf[BEEP_BUF_COUNT] IBSS_ATTR;
|
||||
|
||||
/* Actually output samples into beep_buf */
|
||||
#if defined(CPU_ARM)
|
||||
static FORCE_INLINE void beep_generate(int count)
|
||||
{
|
||||
uint32_t *out = beep_buf;
|
||||
uint32_t s;
|
||||
|
||||
asm volatile (
|
||||
"1: \n"
|
||||
"eor %3, %5, %1, asr #31 \n"
|
||||
"subs %2, %2, #1 \n"
|
||||
"str %3, [%0], #4 \n"
|
||||
"add %1, %1, %4 \n"
|
||||
"bgt 1b \n"
|
||||
: "+r"(out), "+r"(beep_phase), "+r"(count),
|
||||
"=&r"(s)
|
||||
: "r"(beep_step), "r"(beep_amplitude));
|
||||
}
|
||||
#elif defined (CPU_COLDFIRE)
|
||||
static FORCE_INLINE void beep_generate(int count)
|
||||
{
|
||||
uint32_t *out = beep_buf;
|
||||
uint32_t s;
|
||||
|
||||
asm volatile (
|
||||
"1: \n"
|
||||
"move.l %1, %3 \n"
|
||||
"add.l %4, %1 \n"
|
||||
"add.l %3, %3 \n"
|
||||
"subx.l %3, %3 \n"
|
||||
"eor.l %5, %3 \n"
|
||||
"move.l %3, (%0)+ \n"
|
||||
"subq.l #1, %2 \n"
|
||||
"bgt.b 1b \n"
|
||||
: "+a"(out), "+d"(beep_phase), "+d"(count),
|
||||
"=&d"(s)
|
||||
: "r"(beep_step), "d"(beep_amplitude));
|
||||
}
|
||||
#else
|
||||
static FORCE_INLINE void beep_generate(int count)
|
||||
{
|
||||
uint32_t *out = beep_buf;
|
||||
uint32_t amplitude = beep_amplitude;
|
||||
uint32_t step = beep_step;
|
||||
int32_t phase = beep_phase;
|
||||
|
||||
do
|
||||
{
|
||||
*out++ = (phase >> 31) ^ amplitude;
|
||||
phase += step;
|
||||
}
|
||||
while (--count > 0);
|
||||
|
||||
beep_phase = phase;
|
||||
}
|
||||
#endif
|
||||
static int16_t beep_buf[BEEP_BUF_COUNT*2] IBSS_ATTR __attribute__((aligned(4)));
|
||||
|
||||
/* Callback to generate the beep frames - also don't want inlining of
|
||||
call below in beep_play */
|
||||
|
@ -104,8 +55,9 @@ beep_get_more(unsigned char **start, size_t *size)
|
|||
count = MIN(count, BEEP_BUF_COUNT);
|
||||
beep_count -= count;
|
||||
*start = (unsigned char *)beep_buf;
|
||||
*size = count * sizeof(uint32_t);
|
||||
beep_generate(count);
|
||||
*size = count * 2 * sizeof (int16_t);
|
||||
beep_generate((void *)beep_buf, count, &beep_phase,
|
||||
beep_step, beep_amplitude);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -126,7 +78,13 @@ void beep_play(unsigned int frequency, unsigned int duration,
|
|||
beep_phase = 0;
|
||||
beep_step = 0xffffffffu / NATIVE_FREQUENCY * frequency;
|
||||
beep_count = NATIVE_FREQUENCY / 1000 * duration;
|
||||
|
||||
#ifdef BEEP_GENERIC
|
||||
beep_amplitude = amplitude;
|
||||
#else
|
||||
/* Optimized routines do XOR with phase sign bit in both channels at once */
|
||||
beep_amplitude = amplitude | (amplitude << 16); /* Word:|AMP16|AMP16| */
|
||||
#endif
|
||||
|
||||
/* If it fits - avoid cb overhead */
|
||||
unsigned char *start;
|
||||
|
|
39
firmware/asm/arm/beep.c
Normal file
39
firmware/asm/arm/beep.c
Normal file
|
@ -0,0 +1,39 @@
|
|||
/***************************************************************************
|
||||
* __________ __ ___.
|
||||
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
|
||||
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
|
||||
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
|
||||
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
|
||||
* \/ \/ \/ \/ \/
|
||||
* $Id$
|
||||
*
|
||||
* 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.
|
||||
*
|
||||
****************************************************************************/
|
||||
|
||||
/* Actually output samples into beep_buf */
|
||||
static FORCE_INLINE void beep_generate(uint32_t *out, int count,
|
||||
int32_t *phase, uint32_t step,
|
||||
uint32_t amplitude)
|
||||
{
|
||||
uint32_t s;
|
||||
|
||||
asm volatile (
|
||||
"1: \n"
|
||||
"eor %3, %5, %1, asr #31 \n"
|
||||
"subs %2, %2, #1 \n"
|
||||
"str %3, [%0], #4 \n"
|
||||
"add %1, %1, %4 \n"
|
||||
"bgt 1b \n"
|
||||
: "+r"(out), "+r"(*phase), "+r"(count),
|
||||
"=&r"(s)
|
||||
: "r"(step), "r"(amplitude));
|
||||
}
|
51
firmware/asm/beep.c
Normal file
51
firmware/asm/beep.c
Normal file
|
@ -0,0 +1,51 @@
|
|||
/***************************************************************************
|
||||
* __________ __ ___.
|
||||
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
|
||||
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
|
||||
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
|
||||
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
|
||||
* \/ \/ \/ \/ \/
|
||||
* $Id$
|
||||
*
|
||||
* 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.
|
||||
*
|
||||
****************************************************************************/
|
||||
#if defined(CPU_ARM)
|
||||
#include "arm/beep.c"
|
||||
#elif defined (CPU_COLDFIRE)
|
||||
#include "m68k/beep.c"
|
||||
#else /* Generic */
|
||||
|
||||
static FORCE_INLINE void beep_generate(int16_t *out, int count,
|
||||
uint32_t *phase, uint32_t step,
|
||||
int16_t amplitude)
|
||||
{
|
||||
uint32_t ph = *phase;
|
||||
|
||||
do
|
||||
{
|
||||
int16_t amp = amplitude;
|
||||
|
||||
if (ph > UINT32_MAX / 2)
|
||||
amp = -amp;
|
||||
|
||||
*out++ = amp;
|
||||
*out++ = amp;
|
||||
|
||||
ph += step;
|
||||
}
|
||||
while (--count > 0);
|
||||
|
||||
*phase = ph;
|
||||
}
|
||||
|
||||
#define BEEP_GENERIC
|
||||
#endif /* CPU_* */
|
42
firmware/asm/m68k/beep.c
Normal file
42
firmware/asm/m68k/beep.c
Normal file
|
@ -0,0 +1,42 @@
|
|||
/***************************************************************************
|
||||
* __________ __ ___.
|
||||
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
|
||||
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
|
||||
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
|
||||
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
|
||||
* \/ \/ \/ \/ \/
|
||||
* $Id$
|
||||
*
|
||||
* 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.
|
||||
*
|
||||
****************************************************************************/
|
||||
|
||||
/* Actually output samples into beep_buf */
|
||||
static FORCE_INLINE void beep_generate(uint32_t *out, int count,
|
||||
int32_t *phase, uint32_t step,
|
||||
uint32_t amplitude)
|
||||
{
|
||||
uint32_t s;
|
||||
|
||||
asm volatile (
|
||||
"1: \n"
|
||||
"move.l %1, %3 \n"
|
||||
"add.l %4, %1 \n"
|
||||
"add.l %3, %3 \n"
|
||||
"subx.l %3, %3 \n"
|
||||
"eor.l %5, %3 \n"
|
||||
"move.l %3, (%0)+ \n"
|
||||
"subq.l #1, %2 \n"
|
||||
"bgt.b 1b \n"
|
||||
: "+a"(out), "+d"(*phase), "+d"(count),
|
||||
"=&d"(s)
|
||||
: "r"(step), "d"(amplitude));
|
||||
}
|
Loading…
Reference in a new issue