rockbox/firmware/target/arm/ipod/piezo.c

212 lines
5.7 KiB
C
Raw Normal View History

/***************************************************************************
* __________ __ ___.
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
* \/ \/ \/ \/ \/
* $Id$
*
* Copyright (C) 2006-2007 Robert Keevil
*
* 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 "thread.h"
#include "system.h"
#include "kernel.h"
#include "usb.h"
#include "logf.h"
#include "piezo.h"
static long piezo_stack[DEFAULT_STACK_SIZE/sizeof(long)];
static const char piezo_thread_name[] = "piezo";
static struct event_queue piezo_queue;
static unsigned int duration;
static bool beeping;
enum {
Q_PIEZO_BEEP = 1,
Q_PIEZO_BEEP_FOR_TICK,
Q_PIEZO_BEEP_FOR_USEC,
Q_PIEZO_STOP
};
static inline void piezo_hw_init(void)
{
#ifndef SIMULATOR
/*logf("PIEZO: hw_init");*/
outl(inl(0x70000010) & ~0xc, 0x70000010);
outl(inl(0x6000600c) | 0x20000, 0x6000600c); /* enable device */
#endif
}
static void piezo_hw_tick(unsigned int form_and_period)
{
#ifndef SIMULATOR
outl(0x80000000 | form_and_period, 0x7000a000); /* set pitch */
#endif
}
static inline void piezo_hw_stop(void)
{
#ifndef SIMULATOR
outl(0x0, 0x7000a000); /* piezo off */
#endif
}
static void piezo_thread(void)
{
struct queue_event ev;
long piezo_usec_off;
while(1)
{
queue_wait(&piezo_queue, &ev);
switch(ev.id)
{
case Q_PIEZO_BEEP:
piezo_hw_tick((unsigned int)ev.data);
beeping = true;
break;
case Q_PIEZO_BEEP_FOR_TICK:
piezo_hw_tick((unsigned int)ev.data);
beeping = true;
sleep(duration);
if (beeping)
piezo_hw_stop();
beeping = false;
/* remove anything that appeared while sleeping */
queue_clear(&piezo_queue);
break;
case Q_PIEZO_BEEP_FOR_USEC:
piezo_usec_off = USEC_TIMER + duration;
piezo_hw_tick((unsigned int)ev.data);
beeping = true;
while (TIME_BEFORE(USEC_TIMER, piezo_usec_off))
if (duration >= 5000) yield();
if (beeping)
piezo_hw_stop();
beeping = false;
/* remove anything that appeared while sleeping */
queue_clear(&piezo_queue);
break;
case Q_PIEZO_STOP:
if (beeping)
piezo_hw_stop();
beeping = false;
break;
#ifndef SIMULATOR
case SYS_USB_CONNECTED:
/*logf("USB: Piezo core");*/
piezo_hw_stop();
queue_clear(&piezo_queue);
usb_acknowledge(SYS_USB_CONNECTED_ACK);
usb_wait_for_disconnect(&piezo_queue);
break ;
#endif
case SYS_TIMEOUT:
break;
}
yield();
}
}
void piezo_play(unsigned short inv_freq, unsigned char form)
{
queue_post(&piezo_queue, Q_PIEZO_BEEP,
(intptr_t)((unsigned int)form << 16 | inv_freq));
}
void piezo_play_for_tick(unsigned short inv_freq,
unsigned char form, unsigned int dur)
{
duration = dur;
queue_post(&piezo_queue, Q_PIEZO_BEEP_FOR_TICK,
(intptr_t)((unsigned int)form << 16 | inv_freq));
}
void piezo_play_for_usec(unsigned short inv_freq,
unsigned char form, unsigned int dur)
{
duration = dur;
queue_post(&piezo_queue, Q_PIEZO_BEEP_FOR_USEC,
(intptr_t)((unsigned int)form << 16 | inv_freq));
}
void piezo_stop(void)
{
queue_post(&piezo_queue, Q_PIEZO_STOP, 0);
}
void piezo_clear(void)
{
queue_clear(&piezo_queue);
piezo_stop();
}
bool piezo_busy(void)
{
return !queue_empty(&piezo_queue);
}
/* conversion factor based on the following data
period Hz
10 8547
20 4465
30 3024
40 2286
50 1846
60 1537
70 1320
80 1165
90 1030
100 928
someone with better recording/analysing equipment should be able
to get more accurate figures
*/
unsigned int piezo_hz(unsigned int hz)
{
if (hz > 0)
return 91225/hz;
else
return 0;
}
void piezo_init(void)
{
/*logf("PIEZO: init");*/
piezo_hw_init();
queue_init(&piezo_queue, true);
create_thread(piezo_thread, piezo_stack, sizeof(piezo_stack), 0,
piezo_thread_name IF_PRIO(, PRIORITY_REALTIME)
IF_COP(, CPU));
}
void piezo_button_beep(bool beep, bool force)
{
/* old on clickwheel action - piezo_play_for_usec(50, 0x80, 400);
old on button action - piezo_play_for_usec(50, 0x80, 3000); */
if (force)
piezo_clear();
if (queue_empty(&piezo_queue))
{
if (beep)
piezo_play_for_tick(40, 0x80, HZ/5);
else
piezo_play_for_usec(91, 0x80, 4000);
}
}