18c2ecf20Sopenharmony_ci/*
28c2ecf20Sopenharmony_ci * linux/arch/m68k/atari/atasound.c
38c2ecf20Sopenharmony_ci *
48c2ecf20Sopenharmony_ci * ++Geert: Moved almost all stuff to linux/drivers/sound/
58c2ecf20Sopenharmony_ci *
68c2ecf20Sopenharmony_ci * The author of atari_nosound, atari_mksound and atari_microwire_cmd is
78c2ecf20Sopenharmony_ci * unknown. (++roman: That's me... :-)
88c2ecf20Sopenharmony_ci *
98c2ecf20Sopenharmony_ci * This file is subject to the terms and conditions of the GNU General Public
108c2ecf20Sopenharmony_ci * License.  See the file COPYING in the main directory of this archive
118c2ecf20Sopenharmony_ci * for more details.
128c2ecf20Sopenharmony_ci *
138c2ecf20Sopenharmony_ci * 1998-05-31 ++andreas: atari_mksound rewritten to always use the envelope,
148c2ecf20Sopenharmony_ci *			 no timer, atari_nosound removed.
158c2ecf20Sopenharmony_ci *
168c2ecf20Sopenharmony_ci */
178c2ecf20Sopenharmony_ci
188c2ecf20Sopenharmony_ci
198c2ecf20Sopenharmony_ci#include <linux/sched.h>
208c2ecf20Sopenharmony_ci#include <linux/timer.h>
218c2ecf20Sopenharmony_ci#include <linux/major.h>
228c2ecf20Sopenharmony_ci#include <linux/fcntl.h>
238c2ecf20Sopenharmony_ci#include <linux/errno.h>
248c2ecf20Sopenharmony_ci#include <linux/mm.h>
258c2ecf20Sopenharmony_ci#include <linux/module.h>
268c2ecf20Sopenharmony_ci
278c2ecf20Sopenharmony_ci#include <asm/atarihw.h>
288c2ecf20Sopenharmony_ci#include <asm/irq.h>
298c2ecf20Sopenharmony_ci#include <asm/atariints.h>
308c2ecf20Sopenharmony_ci
318c2ecf20Sopenharmony_ci
328c2ecf20Sopenharmony_ci/*
338c2ecf20Sopenharmony_ci * stuff from the old atasound.c
348c2ecf20Sopenharmony_ci */
358c2ecf20Sopenharmony_ci
368c2ecf20Sopenharmony_civoid atari_microwire_cmd (int cmd)
378c2ecf20Sopenharmony_ci{
388c2ecf20Sopenharmony_ci	tt_microwire.mask = 0x7ff;
398c2ecf20Sopenharmony_ci	tt_microwire.data = MW_LM1992_ADDR | cmd;
408c2ecf20Sopenharmony_ci
418c2ecf20Sopenharmony_ci	/* Busy wait for data being completely sent :-( */
428c2ecf20Sopenharmony_ci	while( tt_microwire.mask != 0x7ff)
438c2ecf20Sopenharmony_ci		;
448c2ecf20Sopenharmony_ci}
458c2ecf20Sopenharmony_ciEXPORT_SYMBOL(atari_microwire_cmd);
468c2ecf20Sopenharmony_ci
478c2ecf20Sopenharmony_ci
488c2ecf20Sopenharmony_ci/* PSG base frequency */
498c2ecf20Sopenharmony_ci#define	PSG_FREQ	125000
508c2ecf20Sopenharmony_ci/* PSG envelope base frequency times 10 */
518c2ecf20Sopenharmony_ci#define PSG_ENV_FREQ_10	78125
528c2ecf20Sopenharmony_ci
538c2ecf20Sopenharmony_civoid atari_mksound (unsigned int hz, unsigned int ticks)
548c2ecf20Sopenharmony_ci{
558c2ecf20Sopenharmony_ci	/* Generates sound of some frequency for some number of clock
568c2ecf20Sopenharmony_ci	   ticks.  */
578c2ecf20Sopenharmony_ci	unsigned long flags;
588c2ecf20Sopenharmony_ci	unsigned char tmp;
598c2ecf20Sopenharmony_ci	int period;
608c2ecf20Sopenharmony_ci
618c2ecf20Sopenharmony_ci	local_irq_save(flags);
628c2ecf20Sopenharmony_ci
638c2ecf20Sopenharmony_ci
648c2ecf20Sopenharmony_ci	/* Disable generator A in mixer control.  */
658c2ecf20Sopenharmony_ci	sound_ym.rd_data_reg_sel = 7;
668c2ecf20Sopenharmony_ci	tmp = sound_ym.rd_data_reg_sel;
678c2ecf20Sopenharmony_ci	tmp |= 011;
688c2ecf20Sopenharmony_ci	sound_ym.wd_data = tmp;
698c2ecf20Sopenharmony_ci
708c2ecf20Sopenharmony_ci	if (hz) {
718c2ecf20Sopenharmony_ci	    /* Convert from frequency value to PSG period value (base
728c2ecf20Sopenharmony_ci	       frequency 125 kHz).  */
738c2ecf20Sopenharmony_ci
748c2ecf20Sopenharmony_ci	    period = PSG_FREQ / hz;
758c2ecf20Sopenharmony_ci
768c2ecf20Sopenharmony_ci	    if (period > 0xfff) period = 0xfff;
778c2ecf20Sopenharmony_ci
788c2ecf20Sopenharmony_ci	/* Set generator A frequency to hz.  */
798c2ecf20Sopenharmony_ci	sound_ym.rd_data_reg_sel = 0;
808c2ecf20Sopenharmony_ci	sound_ym.wd_data = period & 0xff;
818c2ecf20Sopenharmony_ci	sound_ym.rd_data_reg_sel = 1;
828c2ecf20Sopenharmony_ci	sound_ym.wd_data = (period >> 8) & 0xf;
838c2ecf20Sopenharmony_ci	if (ticks) {
848c2ecf20Sopenharmony_ci		/* Set length of envelope (max 8 sec).  */
858c2ecf20Sopenharmony_ci		int length = (ticks * PSG_ENV_FREQ_10) / HZ / 10;
868c2ecf20Sopenharmony_ci
878c2ecf20Sopenharmony_ci		if (length > 0xffff) length = 0xffff;
888c2ecf20Sopenharmony_ci		sound_ym.rd_data_reg_sel = 11;
898c2ecf20Sopenharmony_ci		sound_ym.wd_data = length & 0xff;
908c2ecf20Sopenharmony_ci		sound_ym.rd_data_reg_sel = 12;
918c2ecf20Sopenharmony_ci		sound_ym.wd_data = length >> 8;
928c2ecf20Sopenharmony_ci		/* Envelope form: max -> min single.  */
938c2ecf20Sopenharmony_ci		sound_ym.rd_data_reg_sel = 13;
948c2ecf20Sopenharmony_ci		sound_ym.wd_data = 0;
958c2ecf20Sopenharmony_ci		/* Use envelope for generator A.  */
968c2ecf20Sopenharmony_ci		sound_ym.rd_data_reg_sel = 8;
978c2ecf20Sopenharmony_ci		sound_ym.wd_data = 0x10;
988c2ecf20Sopenharmony_ci	} else {
998c2ecf20Sopenharmony_ci		/* Set generator A level to maximum, no envelope.  */
1008c2ecf20Sopenharmony_ci		sound_ym.rd_data_reg_sel = 8;
1018c2ecf20Sopenharmony_ci		sound_ym.wd_data = 15;
1028c2ecf20Sopenharmony_ci	}
1038c2ecf20Sopenharmony_ci	/* Turn on generator A in mixer control.  */
1048c2ecf20Sopenharmony_ci	sound_ym.rd_data_reg_sel = 7;
1058c2ecf20Sopenharmony_ci	tmp &= ~1;
1068c2ecf20Sopenharmony_ci	sound_ym.wd_data = tmp;
1078c2ecf20Sopenharmony_ci	}
1088c2ecf20Sopenharmony_ci	local_irq_restore(flags);
1098c2ecf20Sopenharmony_ci}
110