/************************************************************************** fmout.cc - class fmOut which handles the /dev/sequencer device for fm synths This file is part of LibKMid 0.9.5 Copyright (C) 1998,99,2000 Antonio Larrosa Jimenez LibKMid's homepage : http://www.arrakis.es/~rlarrosa/libkmid.html This library is free software; you can redistribute it and/or modify it under the terms of the GNU Library General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This library 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 Library General Public License for more details. You should have received a copy of the GNU Library General Public License along with this library; see the file COPYING.LIB. If not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. Send comments and bug fixes to Antonio Larrosa <larrosa@kde.org> ***************************************************************************/ #include "fmout.h" #include <unistd.h> #include <fcntl.h> #include <stdio.h> #include "sndcard.h" #include <sys/ioctl.h> #include <errno.h> #include <string.h> #include <sys/param.h> #include <stdlib.h> #include <limits.h> #include "midispec.h" #ifdef HAVE_CONFIG_H #include <config.h> #endif SEQ_USE_EXTBUF(); FMOut::FMOut( int d, int total ) { seqfd = -1; devicetype = KMID_FM; device = d; _ok = 1; // Put opl=3 for opl/3 (better quality/ 6 voices) // or opl=2 for fm output (less quality/ 18 voices, which is better imho) : opl = 2; // But be aware that opl=3 is not intended to be fully supported by now nvoices = total; vm = new VoiceManager (nvoices); } FMOut::~FMOut() { closeDev(); delete vm; if (deleteFMPatchesDirectory) { free((char *)FMPatchesDirectory); deleteFMPatchesDirectory = 0; FMPatchesDirectory="/etc"; } } void FMOut::openDev (int sqfd) { #ifdef HAVE_OSS_SUPPORT _ok=1; seqfd = sqfd; //vm->clearLists(); if ( seqfd == -1 ) { printfdebug("ERROR: Could not open /dev/sequencer\n"); return; } loadFMPatches(); #endif } void FMOut::closeDev (void) { if (!ok()) return; vm->clearLists(); //if (seqfd>=0) close(seqfd); seqfd = -1; } void FMOut::initDev (void) { #ifdef HAVE_OSS_SUPPORT int chn; if (!ok()) return; uchar gm_reset[5]={0x7e, 0x7f, 0x09, 0x01, 0xf7}; sysex(gm_reset, sizeof(gm_reset)); for (chn=0;chn<16;chn++) { chnmute[chn]=0; chnPatchChange(chn,0); chnPressure(chn,127); chnPitchBender(chn, 0x00, 0x40); chnController(chn, CTL_MAIN_VOLUME,127); chnController(chn, CTL_EXT_EFF_DEPTH, 0); chnController(chn, CTL_CHORUS_DEPTH, 0); chnController(chn, 0x4a, 127); } if (opl==3) ioctl(seqfd, SNDCTL_FM_4OP_ENABLE, &device); SEQ_VOLUME_MODE(device,VOL_METHOD_LINEAR); for (int i = 0; i < nvoices; i++) { SEQ_CONTROL(device, i, SEQ_VOLMODE, VOL_METHOD_LINEAR); SEQ_STOP_NOTE(device, i, vm->note(i), 64); } #endif } void FMOut::loadFMPatches(void) { #ifdef HAVE_OSS_SUPPORT char patchesfile[strlen(FMPatchesDirectory)+7+1]; char drumsfile[strlen(FMPatchesDirectory)+9+1]; int size; struct sbi_instrument instr; char tmp[60]; int i,j; for ( i=0; i<256; i++ ) patchloaded[i] = 0; int stereoeffect=rand()%3; FILE *fh; int datasize; if (opl==3) { snprintf(patchesfile, sizeof(patchesfile), "%s/std.o3",FMPatchesDirectory); size=60; } else { snprintf(patchesfile, sizeof(patchesfile), "%s/std.sb",FMPatchesDirectory); size=52; } fh=fopen(patchesfile,"rb"); if (fh==NULL) return; for (i=0;i<128;i++) { fread(tmp,size,1,fh); patchloaded[i]=1; instr.key = ((strncmp(tmp, "4OP", 3) == 0))? OPL3_PATCH : FM_PATCH; datasize = (strncmp(tmp, "4OP", 3) == 0)? 22 : 11; instr.device=device; instr.channel = i; // Let's get some stereo effect ... tmp[46] = (tmp[46] & 0xcf) | ((++stereoeffect)<<4); stereoeffect=stereoeffect%3; for (j=0; j<22; j++) instr.operators[j] = tmp[j+36]; SEQ_WRPATCH(&instr,sizeof(instr)); } fclose(fh); if (opl==3) { snprintf(drumsfile, sizeof(drumsfile), "%s/drums.o3",FMPatchesDirectory); } else { snprintf(drumsfile, sizeof(drumsfile), "%s/drums.sb",FMPatchesDirectory); } fh=fopen(drumsfile,"rb"); if (fh==NULL) return; for (i=128;i<175;i++) { fread(tmp,size,1,fh); patchloaded[i]=1; instr.key = (strncmp(tmp, "4OP", 3) == 0)? OPL3_PATCH : FM_PATCH; datasize = (strncmp(tmp, "4OP", 3) == 0)? 22 : 11; instr.device=device; instr.channel = i; // Let's get some stereo effect ... tmp[46] = (tmp[46] & 0xcf) | ((++stereoeffect)<<4); stereoeffect=stereoeffect%3; for (j=0; j<22; j++) instr.operators[j] = tmp[j+36]; SEQ_WRPATCH(&instr,sizeof(instr)); } fclose(fh); #ifdef FMOUTDEBUG printfdebug("Patches loaded\n"); #endif #endif } int FMOut::patch(int p) { if (patchloaded[p]==1) return p; #ifdef FMOUTDEBUG printfdebug("Not loaded %d!\n",p); #endif p=0; while ((p<256)&&(patchloaded[p]==0)) p++; return p; } void FMOut::noteOn (uchar chn, uchar note, uchar vel) { if (vel==0) { noteOff(chn,note,vel); } else { if (chn==PERCUSSION_CHANNEL) { if (patchloaded[note+128]==0) return; else if (patchloaded[chnpatch[chn]]==0) return; } int v=vm->allocateVoice(chn,note); int p; if (chn==PERCUSSION_CHANNEL) SEQ_SET_PATCH(device,v ,p=patch(note+128)) else SEQ_SET_PATCH(device,v ,p=map->patch(chn,chnpatch[chn])); SEQ_BENDER(device, v, chnbender[chn]); SEQ_START_NOTE(device, v, note, vel); // SEQ_CONTROL(device, v, CTL_MAIN_VOLUME, chncontroller[chn][CTL_MAIN_VOLUME]); SEQ_CHN_PRESSURE(device, v , chnpressure[chn]); } #ifdef FMOUTDEBUG printfdebug("Note ON >\t chn : %d\tnote : %d\tvel: %d\n",chn,note,vel); #endif } void FMOut::noteOff (uchar chn, uchar note, uchar vel) { int i; vm->initSearch(); while ((i=vm->search(chn,note))!=-1) { SEQ_STOP_NOTE(device, i, note, vel); vm->deallocateVoice(i); } #ifdef FMOUTDEBUG printfdebug("Note OFF >\t chn : %d\tnote : %d\tvel: %d\n",chn,note,vel); #endif } void FMOut::keyPressure (uchar chn, uchar note, uchar vel) { int i; vm->initSearch(); while ((i=vm->search(chn,note))!=-1) SEQ_KEY_PRESSURE(device, i, note,vel); } void FMOut::chnPatchChange (uchar chn, uchar patch) { if (chn==PERCUSSION_CHANNEL) return; int i; vm->initSearch(); while ((i=vm->search(chn))!=-1) SEQ_SET_PATCH(device,i,map->patch(chn,patch)); chnpatch[chn]=patch; } void FMOut::chnPressure (uchar chn, uchar vel) { int i; vm->initSearch(); while ((i=vm->search(chn))!=-1) SEQ_CHN_PRESSURE(device, i , vel); chnpressure[chn]=vel; } void FMOut::chnPitchBender(uchar chn,uchar lsb, uchar msb) { chnbender[chn]=((int)msb<<7) | (lsb & 0x7F); int i; vm->initSearch(); while ((i=vm->search(chn))!=-1) SEQ_BENDER(device, i, chnbender[chn]); } void FMOut::chnController (uchar chn, uchar ctl, uchar v) { if ((ctl==11)||(ctl==7)) { v=(v*volumepercentage)/100; if (v>127) v=127; } int i; vm->initSearch(); while ((i=vm->search(chn))!=-1) SEQ_CONTROL(device, i, ctl, v); chncontroller[chn][ctl]=v; } void FMOut::sysex(uchar *, ulong ) { } void FMOut::setFMPatchesDirectory(const char *dir) { if ((dir==NULL)||(dir[0]==0)) return; if (deleteFMPatchesDirectory) free((char *)FMPatchesDirectory); FMPatchesDirectory = strdup(dir); deleteFMPatchesDirectory=1; } void FMOut::setVolumePercentage ( int i ) { #ifdef HAVE_OSS_SUPPORT int fd=open("/dev/mixer0",O_RDWR,0); if (fd==-1) return; int a=i*255/100; if (a>255) a=255; a=(a<<8) | a; if (ioctl(fd,MIXER_WRITE(SOUND_MIXER_SYNTH),&a) == -1) printfdebug("ERROR writing to mixer\n"); close(fd); #endif volumepercentage=i; } const char *FMOut::FMPatchesDirectory = "/etc"; int FMOut::deleteFMPatchesDirectory = 0;