145 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			145 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /**
 | |
|  * Furnace Tracker - multi-system chiptune tracker
 | |
|  * Copyright (C) 2021-2022 tildearrow and contributors
 | |
|  *
 | |
|  * 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 <SDL_audio.h>
 | |
| #include <string.h>
 | |
| #include <vector>
 | |
| #include "../ta-log.h"
 | |
| #include "sdl.h"
 | |
| 
 | |
| void taSDLProcess(void* inst, unsigned char* buf, int nframes) {
 | |
|   TAAudioSDL* in=(TAAudioSDL*)inst;
 | |
|   in->onProcess(buf,nframes);
 | |
| }
 | |
| 
 | |
| void TAAudioSDL::onProcess(unsigned char* buf, int nframes) {
 | |
|   if (audioProcCallback!=NULL) {
 | |
|     if (midiIn!=NULL) midiIn->gather();
 | |
|     audioProcCallback(audioProcCallbackUser,inBufs,outBufs,desc.inChans,desc.outChans,desc.bufsize);
 | |
|   }
 | |
|   float* fbuf=(float*)buf;
 | |
|   for (size_t j=0; j<desc.bufsize; j++) {
 | |
|     for (size_t i=0; i<desc.outChans; i++) {
 | |
|       fbuf[j*desc.outChans+i]=outBufs[i][j];
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| void* TAAudioSDL::getContext() {
 | |
|   return (void*)∾
 | |
| }
 | |
| 
 | |
| bool TAAudioSDL::quit() {
 | |
|   if (!initialized) return false;
 | |
| 
 | |
|   SDL_CloseAudioDevice(ai);
 | |
| 
 | |
|   if (running) {
 | |
|     running=false;
 | |
|   }
 | |
| 
 | |
|   for (int i=0; i<desc.outChans; i++) {
 | |
|     delete[] outBufs[i];
 | |
|   }
 | |
| 
 | |
|   delete[] outBufs;
 | |
|   
 | |
|   initialized=false;
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| bool TAAudioSDL::setRun(bool run) {
 | |
|   if (!initialized) return false;
 | |
|   SDL_PauseAudioDevice(ai,!run);
 | |
|   running=run;
 | |
|   return running;
 | |
| }
 | |
| 
 | |
| std::vector<String> TAAudioSDL::listAudioDevices() {
 | |
|   std::vector<String> ret;
 | |
|   if (!audioSysStarted) {
 | |
|     if (SDL_Init(SDL_INIT_AUDIO)<0) {
 | |
|       logE("could not initialize SDL to list audio devices");
 | |
|     } else {
 | |
|       audioSysStarted=true;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   int count=SDL_GetNumAudioDevices(false);
 | |
|   if (count<0) return ret;
 | |
| 
 | |
|   for (int i=0; i<count; i++) {
 | |
|     const char* devName=SDL_GetAudioDeviceName(i,false);
 | |
|     if (devName!=NULL) {
 | |
|       ret.push_back(String(devName));
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return ret;
 | |
| }
 | |
| 
 | |
| bool TAAudioSDL::init(TAAudioDesc& request, TAAudioDesc& response) {
 | |
|   if (initialized) {
 | |
|     logE("audio already initialized");
 | |
|     return false;
 | |
|   }
 | |
|   if (!audioSysStarted) {
 | |
|     if (SDL_Init(SDL_INIT_AUDIO)<0) {
 | |
|       logE("could not initialize SDL");
 | |
|       return false;
 | |
|     }
 | |
|     audioSysStarted=true;
 | |
|   }
 | |
| 
 | |
|   desc=request;
 | |
|   desc.outFormat=TA_AUDIO_FORMAT_F32;
 | |
| 
 | |
|   ac.freq=desc.rate;
 | |
|   ac.format=AUDIO_F32;
 | |
|   ac.channels=desc.outChans;
 | |
|   ac.samples=desc.bufsize;
 | |
|   ac.callback=taSDLProcess;
 | |
|   ac.userdata=this;
 | |
| 
 | |
|   ai=SDL_OpenAudioDevice(request.deviceName.empty()?NULL:request.deviceName.c_str(),0,&ac,&ar,SDL_AUDIO_ALLOW_FREQUENCY_CHANGE);
 | |
|   if (ai==0) {
 | |
|     logE("could not open audio device: %s",SDL_GetError());
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   desc.deviceName=request.deviceName;
 | |
|   desc.name="";
 | |
|   desc.rate=ar.freq;
 | |
|   desc.inChans=0;
 | |
|   desc.outChans=ar.channels;
 | |
|   desc.bufsize=ar.samples;
 | |
|   desc.fragments=1;
 | |
| 
 | |
|   if (desc.outChans>0) {
 | |
|     outBufs=new float*[desc.outChans];
 | |
|     for (int i=0; i<desc.outChans; i++) {
 | |
|       outBufs[i]=new float[desc.bufsize];
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   response=desc;
 | |
|   initialized=true;
 | |
|   return true;
 | |
| }
 | 
