Mercurial > audlegacy-plugins
view src/metronom/metronom.c @ 3182:631d217913e0
alsa-ng: Fix arithmetic overflow on 32-bit systems.
author | John Lindgren <john.lindgren@tds.net> |
---|---|
date | Fri, 12 Jun 2009 09:17:09 -0400 |
parents | 3134a0987162 |
children |
line wrap: on
line source
/* * Copyright 2000 Martin Strau? <mys@faveve.uni-stuttgart.de> * * * 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 program 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ #include "config.h" #include <audlegacy/plugin.h> #include <audlegacy/output.h> #include <audlegacy/i18n.h> #include <glib.h> #include <string.h> #define MIN_BPM 1 #define MAX_BPM 512 static InputPlugin metronom_ip; static gboolean going; static gboolean audio_error; static GThread *play_thread; struct metronom_struct { gint bpm; gint num; gint den; gint id; }; typedef struct metronom_struct metronom_t; #define tact_id_max 12 gint tact_id[tact_id_max][2]= { {1,1}, {2,2}, {3,2}, {4,2}, {2,4}, {3,4}, {4,4}, {6,4}, {2,8}, {3,8}, {4,8}, {6,8} }; #define tact_form_max 8 gdouble tact_form[tact_id_max][tact_form_max]= { {1.0,0.0,0.0,0.0,0.0,0.0,0.0,0.0}, {1.0,0.5,0.0,0.0,0.0,0.0,0.0,0.0}, {1.0,0.5,0.5,0.0,0.0,0.0,0.0,0.0}, {1.0,0.5,0.6,0.5,0.0,0.0,0.0,0.0}, {1.0,0.5,0.0,0.0,0.0,0.0,0.0,0.0}, {1.0,0.5,0.5,0.0,0.0,0.0,0.0,0.0}, {1.0,0.5,0.6,0.5,0.0,0.0,0.0,0.0}, {1.0,0.5,0.5,0.6,0.5,0.5,0.0,0.0}, {1.0,0.5,0.0,0.0,0.0,0.0,0.0,0.0}, {1.0,0.5,0.5,0.0,0.0,0.0,0.0,0.0}, {1.0,0.5,0.6,0.5,0.0,0.0,0.0,0.0}, {1.0,0.5,0.5,0.6,0.5,0.5,0.0,0.0} }; static void metronom_init(void) { aud_uri_set_plugin("tact://", &metronom_ip); } static void metronom_about(void) { static GtkWidget *box; box = audacious_info_dialog( _("About Metronom"), _("A Tact Generator by Martin Strauss <mys@faveve.uni-stuttgart.de>\n\nTo use it, add a URL: tact://beats*num/den\ne.g. tact://77 to play 77 beats per minute\nor tact://60*3/4 to play 60 bpm in 3/4 tacts"), _("Ok"), FALSE, NULL, NULL); gtk_signal_connect(GTK_OBJECT(box), "destroy", GTK_SIGNAL_FUNC(gtk_widget_destroyed), &box); } static int metronom_is_our_file(char *filename) { if (!strncmp(filename, "tact://", 7)) return TRUE; return FALSE; } #define BUF_SAMPLES 512 #define BUF_BYTES BUF_SAMPLES * 2 #define MAX_AMPL (GINT16_TO_LE((1<<15) - 1)) static void play_loop(void *arg) { gint16 data[BUF_SAMPLES]; InputPlayback *playback = arg; metronom_t *pmetronom=(metronom_t *)playback->data; gint i; gint16 t = 0,tact; gint16 datagoal = 0; gint16 datamiddle = 0; gint16 datacurrent = datamiddle; gint16 datalast = datamiddle; gint16 data_form[tact_form_max]; gint num; tact = 60*44100/pmetronom->bpm; /* prepare weighted amplitudes */ for(num=0;num<pmetronom->num;num++){ data_form[num]=MAX_AMPL*tact_form[pmetronom->id][num]; } num=0; while (going) { for (i = 0; i < BUF_SAMPLES; i++){ if(t==tact){ t=0; datagoal = data_form[num]; } else if(t==10) { datagoal = -data_form[num]; } else if(t==25) { datagoal = data_form[num]; /* circle through weighted amplitudes */ num++; if(num==pmetronom->num)num=0; } /* makes curve a little bit smoother */ data[i]=(datalast+datacurrent+datagoal)/3; datalast=datacurrent; datacurrent=data[i]; if(t > 35) datagoal=(datamiddle+7*datagoal)/8; t++; } if (going) playback->pass_audio(playback, FMT_S16_LE, 1, BUF_BYTES, data, &going); } /* Make sure the output plugin stops prebuffering */ playback->output->buffer_free(); playback->output->buffer_free(); } static void metronom_play(InputPlayback *playback) { char *filename = playback->filename; gchar *name; size_t count; metronom_t *pmetronom; gint flag,id; pmetronom=(metronom_t *)g_malloc(sizeof(metronom_t)); if(!pmetronom)return; count=sscanf(filename, "tact://%d*%d/%d", &pmetronom->bpm,&pmetronom->num,&pmetronom->den); if (count != 1 && count !=3)return; if(!(pmetronom->bpm >= MIN_BPM && pmetronom->bpm <= MAX_BPM))return; pmetronom->id=0; if(count==1){ pmetronom->num=1; pmetronom->den=1; } else { if(pmetronom->num==0 || pmetronom->den==0)return; flag=FALSE; for(id=0;(id<tact_id_max && (!flag));id++){ if(pmetronom->num==tact_id[id][0] && pmetronom->den==tact_id[id][1]){ flag=TRUE; pmetronom->id=id; } } if(!flag)return; } going = TRUE; audio_error = FALSE; if (playback->output->open_audio(FMT_S16_LE, 44100, 1) == 0) { audio_error = TRUE; going = FALSE; return; } if(pmetronom->num==1 && pmetronom->den==1){ name = g_strdup_printf(_("Tact generator: %d bpm"), pmetronom->bpm); } else { name = g_strdup_printf(_("Tact generator: %d bpm %d/%d"), pmetronom->bpm,pmetronom->num,pmetronom->den); } playback->set_params(playback, name, -1, 16 * 44100, 44100, 1); g_free(name); playback->data = pmetronom; play_thread = g_thread_self(); playback->set_pb_ready(playback); play_loop(playback); } static void metronom_stop(InputPlayback *playback) { if (going) { going = FALSE; g_thread_join(play_thread); playback->output->close_audio(); } } static void metronom_pause(InputPlayback *playback, short paused) { playback->output->pause(paused); } static int metronom_get_time(InputPlayback *playback) { if (audio_error) return -2; if (!going && !playback->output->buffer_playing()) return -1; return playback->output->output_time(); } static void metronom_song_info(char *filename, char **title, int *length) { metronom_t metronom; metronom_t *pmetronom=&metronom; size_t count; gint flag,id; *length = -1; *title = NULL; count=sscanf(filename, "tact://%d*%d/%d", &pmetronom->bpm,&pmetronom->num,&pmetronom->den); if (count != 1 && count !=3)return; if(!(pmetronom->bpm >= MIN_BPM && pmetronom->bpm <= MAX_BPM))return; if (count == 1) { pmetronom->num=1; pmetronom->den=1; pmetronom->id=0; } else { if(pmetronom->num==0 || pmetronom->den==0)return; flag=FALSE; for(id=0;(id<tact_id_max && (!flag));id++){ if(pmetronom->num==tact_id[id][0] && pmetronom->den==tact_id[id][1])flag=TRUE; } if(!flag)return; else pmetronom->id=id; } if(pmetronom->num==1 && pmetronom->den==1){ *title = g_strdup_printf(_("Tact generator: %d bpm"), pmetronom->bpm); } else { *title = g_strdup_printf(_("Tact generator: %d bpm %d/%d"), pmetronom->bpm,pmetronom->num,pmetronom->den); } } static InputPlugin metronom_ip = { .description = "Tact Generator", .init = metronom_init, .about = metronom_about, .is_our_file = metronom_is_our_file, .play_file = metronom_play, .stop = metronom_stop, .pause = metronom_pause, .get_time = metronom_get_time, .get_song_info = metronom_song_info, }; InputPlugin *metronom_iplist[] = { &metronom_ip, NULL }; DECLARE_PLUGIN(metronom, NULL, NULL, metronom_iplist, NULL, NULL, NULL, NULL, NULL);