/* ScummVM - Graphic Adventure Engine
 *
 * ScummVM is the legal property of its developers, whose names
 * are too numerous to list here. Please refer to the COPYRIGHT
 * file distributed with this source distribution.
 *
 * 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.
 *
 * $URL$
 * $Id$
 *
 */

/* PC speaker software sequencer for FreeSCI */

#include "sci/sfx/softseq.h"
#include "sci/include/sci_midi.h"

namespace Sci {

#define FREQUENCY 94020

static int volume = 0x0600;
static int note = 0; /* Current halftone, or 0 if off */
static int freq_count = 0;

extern sfx_softseq_t sfx_softseq_pcspeaker;
/* Forward-declare the sequencer we are defining here */


static int sps_set_option(sfx_softseq_t *self, const char *name, const char *value) {
	return SFX_ERROR;
}

static int sps_init(sfx_softseq_t *self, byte *patch, int patch_len, byte *patch2,
         int patch2_len) {
	return SFX_OK;
}

static void sps_exit(sfx_softseq_t *self) {
}

static void sps_event(sfx_softseq_t *self, byte command, int argc, byte *argv) {
#if 0
	fprintf(stderr, "Note [%02x : %02x %02x]\n", command,  argc ? argv[0] : 0, (argc > 1) ? argv[1] : 0);
#endif

	switch (command & 0xf0) {

	case 0x80:
		if (argv[0] == note)
			note = 0;
		break;

	case 0x90:
		if (!argv[1]) {
			if (argv[0] == note)
				note = 0;
		} else
			note = argv[0];
		/* Ignore velocity otherwise; we just use the global one */
		break;

	case 0xb0:
		if (argv[1] == SCI_MIDI_CHANNEL_NOTES_OFF)
			note = 0;
		break;

	default:
#if DEBUG
		fprintf(stderr, "[SFX:PCM-PC] Unused MIDI command %02x %02x %02x\n", command, argc ? argv[0] : 0, (argc > 1) ? argv[1] : 0);
#endif
		break; /* ignore */
	}
}

#define BASE_NOTE 129	/* A10 */
#define BASE_OCTAVE 10	/* A10, as I said */

static int freq_table[12] = { /* A4 is 440Hz, halftone map is x |-> ** 2^(x/12) */
	28160, /* A10 */
	29834,
	31608,
	33488,
	35479,
	37589,
	39824,
	42192,
	44701,
	47359,
	50175,
	53159
};


void sps_poll(sfx_softseq_t *self, byte *dest, int len) {
	int halftone_delta = note - BASE_NOTE;
	int oct_diff = ((halftone_delta + BASE_OCTAVE * 12) / 12) - BASE_OCTAVE;
	int halftone_index = (halftone_delta + (12 * 100)) % 12 ;
	int freq = (!note) ? 0 : freq_table[halftone_index] / (1 << (-oct_diff));
	gint16 *buf = (gint16 *) dest;

	int i;
	for (i = 0; i < len; i++) {
		if (note) {
			freq_count += freq;
			while (freq_count >= (FREQUENCY << 1))
				freq_count -= (FREQUENCY << 1);

			if (freq_count - freq < 0) {
				/* Unclean rising edge */
				int l = volume << 1;
				buf[i] = -volume + (l * freq_count) / freq;
			} else if (freq_count >= FREQUENCY
			           && freq_count - freq < FREQUENCY) {
				/* Unclean falling edge */
				int l = volume << 1;
				buf[i] = volume - (l * (freq_count - FREQUENCY)) / freq;
			} else {
				if (freq_count < FREQUENCY)
					buf[i] = volume;
				else
					buf[i] = -volume;
			}
		} else
			buf[i] = 0;
	}

}

void sps_volume(sfx_softseq_t *self, int new_volume) {
	volume = new_volume << 4;
}

void sps_allstop(sfx_softseq_t *self) {
	note = 0;
}

sfx_softseq_t sfx_softseq_pcspeaker = {
	"pc-speaker",
	"0.3",
	sps_set_option,
	sps_init,
	sps_exit,
	sps_volume,
	sps_event,
	sps_poll,
	sps_allstop,
	NULL,
	SFX_SEQ_PATCHFILE_NONE,
	SFX_SEQ_PATCHFILE_NONE,
	0x20,  /* PC speaker channel only */
	0, /* No rhythm channel */
	1, /* # of voices */
	{FREQUENCY, SFX_PCM_MONO, SFX_PCM_FORMAT_S16_NATIVE}
};

} // End of namespace Sci