1
0
mirror of git://projects.qi-hardware.com/openwrt-xburst.git synced 2025-01-26 12:01:06 +02:00
blogic e1c9572b15 added libjson-c. added driver, webinterface and userspace daemon for the
fonera mp3-hack


git-svn-id: svn://svn.openwrt.org/openwrt/trunk@8509 3c298f89-4303-0410-b956-a3cf2f4a3e73
2007-08-27 20:10:35 +00:00

205 lines
6.3 KiB
C

/*
* FOXMP3
* Copyright (c) 2006 acmesystems.it - john@acmesystems.it
*
* 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., 59 Temple Place, Suite 330, Boston, MA02111-1307USA
*
* Feedback, Bugs... info@acmesystems.it
*
*/
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/poll.h>
#include <signal.h>
#include "lib/mp3.h"
STATE states[MAX_STATE_COUNT];
static EVENT events[MAX_EVENT_COUNT];
static TRANSITION transitions[MAX_TRANSITION_COUNT];
static int transition_count = 0;
int state_current = MP3_STATE_STARTUP;
static int state_previous = MP3_STATE_NONE;
static int shutdown = 0;
void state_setup(void){
EVENT_ADD(MP3_EVENT_TIMEOUT, "Timeout");
EVENT_ADD(MP3_EVENT_FILE, "File");
EVENT_ADD(MP3_EVENT_STREAM, "Stream");
EVENT_ADD(MP3_EVENT_STOP, "Stop");
EVENT_ADD(MP3_EVENT_ERROR, "Error");
EVENT_ADD(MP3_EVENT_SHUTDOWN, "Shutdown");
EVENT_ADD(MP3_EVENT_END, "End");
STATE_ADD(MP3_STATE_STARTUP, "Startup", state_startup_enter, NULL);
STATE_ADD(MP3_STATE_IDLE, "Idle", state_idle_enter, NULL);
STATE_ADD(MP3_STATE_FILE_START, "File startup", state_file_startup_enter, state_file_startup_leave);
STATE_ADD(MP3_STATE_FILE_HANDLE, "File handle", state_file_handle_enter, state_file_handle_leave);
STATE_ADD(MP3_STATE_STREAM_START, "Stream start", state_stream_startup_enter, state_stream_startup_leave);
STATE_ADD(MP3_STATE_STREAM_HANDLE, "Stream handle", state_stream_handle_enter, state_stream_handle_leave);
STATE_ADD(MP3_STATE_ERROR, "Error", state_error_enter, NULL);
STATE_ADD(MP3_STATE_SHUTDOWN, "Shutdown", state_shutdown_enter, NULL);
TRANSITION_ADD(MP3_STATE_STARTUP, MP3_EVENT_TIMEOUT, MP3_STATE_IDLE);
TRANSITION_ADD(MP3_STATE_IDLE, MP3_EVENT_TIMEOUT, MP3_STATE_IDLE);
TRANSITION_ADD(MP3_STATE_IDLE, MP3_EVENT_FILE, MP3_STATE_FILE_START);
TRANSITION_ADD(MP3_STATE_IDLE, MP3_EVENT_STREAM, MP3_STATE_STREAM_START);
TRANSITION_ADD(MP3_STATE_FILE_START, MP3_EVENT_TIMEOUT, MP3_STATE_FILE_HANDLE);
TRANSITION_ADD(MP3_STATE_FILE_START, MP3_EVENT_ERROR, MP3_STATE_ERROR);
TRANSITION_ADD(MP3_STATE_FILE_HANDLE, MP3_EVENT_TIMEOUT, MP3_STATE_FILE_HANDLE);
TRANSITION_ADD(MP3_STATE_FILE_HANDLE, MP3_EVENT_END, MP3_STATE_IDLE);
TRANSITION_ADD(MP3_STATE_FILE_HANDLE, MP3_EVENT_ERROR, MP3_STATE_ERROR);
TRANSITION_ADD(MP3_STATE_FILE_HANDLE, MP3_EVENT_FILE, MP3_STATE_FILE_START);
TRANSITION_ADD(MP3_STATE_FILE_HANDLE, MP3_EVENT_STOP, MP3_STATE_IDLE);
TRANSITION_ADD(MP3_STATE_FILE_HANDLE, MP3_EVENT_STREAM, MP3_STATE_STREAM_START);
TRANSITION_ADD(MP3_STATE_STREAM_START, MP3_EVENT_TIMEOUT, MP3_STATE_STREAM_HANDLE);
TRANSITION_ADD(MP3_STATE_STREAM_START, MP3_EVENT_ERROR, MP3_STATE_ERROR);
TRANSITION_ADD(MP3_STATE_STREAM_HANDLE, MP3_EVENT_TIMEOUT, MP3_STATE_STREAM_HANDLE);
TRANSITION_ADD(MP3_STATE_STREAM_HANDLE, MP3_EVENT_STOP, MP3_STATE_IDLE);
TRANSITION_ADD(MP3_STATE_STREAM_HANDLE, MP3_EVENT_ERROR, MP3_STATE_ERROR);
TRANSITION_ADD(MP3_STATE_STREAM_HANDLE, MP3_EVENT_FILE, MP3_STATE_FILE_START);
TRANSITION_ADD(MP3_STATE_STREAM_HANDLE, MP3_EVENT_STREAM, MP3_STATE_STREAM_START);
TRANSITION_ADD(MP3_STATE_ERROR, MP3_EVENT_TIMEOUT, MP3_STATE_IDLE);
TRANSITION_ADD(MP3_STATE_DEFAULT, MP3_EVENT_ERROR, MP3_STATE_ERROR);
TRANSITION_ADD(MP3_STATE_DEFAULT, MP3_EVENT_SHUTDOWN, MP3_STATE_SHUTDOWN);
state_startup_enter(0, 0, NULL);
};
void state_transition(int transition, int event, EVENT_PARAM *param){
state_previous = state_current;
state_current = transitions[transition].new_state;
if(state_previous != state_current){
printf("\n -- Transition %s -> %s\n",
states[state_previous].name,
states[state_current].name);
};
if(states[state_previous].leave != NULL)
states[state_previous].leave(state_current, event);
if(states[state_current].enter != NULL)
states[state_current].enter(state_previous, event, param);
if(param){
if(param->text){
free(param->text);
};
free(param);
};
};
void state_event(int event, EVENT_PARAM *param){
int i = 0;
if(event != MP3_EVENT_TIMEOUT){
printf("\n -- Got event %s (%d) in state %s (%d)\n",
events[event].name,
event,
states[state_current].name,
state_current);
};
for(i = 0; i < transition_count; i++){
if((transitions[i].old_state == state_current)
&& (transitions[i].event == event)){
state_transition(i, event, param);
return;
};
};
for(i = 0; i < transition_count; i++){
if((transitions[i].old_state == MP3_STATE_DEFAULT)
&& (transitions[i].event == event)){
state_transition(i, event, param);
return;
};
};
if(param){
if(param->text){
free(param->text);
};
free(param);
};
printf("Event %s not handled in state %s\n", events[event].name, states[state_current].name);
};
void state_mainloop(void){
while(1){
if(shutdown){
state_event(MP3_EVENT_SHUTDOWN, NULL);
};
mp3_nix_socket_handle();
mp3_tcp_socket_handle();
if(state_current == MP3_STATE_IDLE){
poll(0, 0, 200);
} else {
poll(0, 0, 1);
};
state_event(MP3_EVENT_TIMEOUT, NULL);
};
};
EVENT_PARAM* state_new_event(unsigned char *text, int numeric){
EVENT_PARAM *p = (EVENT_PARAM*)malloc(sizeof(EVENT_PARAM));
if(text){
p->text = strdup(text);
} else {
p->text = NULL;
};
p->numeric = numeric;
return p;
};
void sig_handler(int sig){
shutdown = 1;
};
int main(int argc, char **argv) {
unsigned char daemonize = 1;
printf("\n\n");
printf("ALPMP3 - MP3 Statemachine v1.0\n");
if(argc > 1){
if(!strcmp(argv[1], "--no-daemon")){
daemonize = 0;
} else {
printf("I am now going to daemonize. If you want me to stay in ");
printf("the foreground, add the parameter --no-daemon.\n");
};
};
if(daemonize){
daemon(0, 0);
printf("----------------------------------------\n\n");
printf("Made by openwrt.org (blogic@openwrt.org)\n");
printf("\n");
};
signal(SIGINT, sig_handler);
state_setup();
state_mainloop();
return 0;
}