[Csnd] having problems using csound spin with jack
Date | 2011-12-22 07:09 |
From | Iain Duncan |
Subject | [Csnd] having problems using csound spin with jack |
First, thanks to Steven and Rory's help, I've gotten tantalizingly close. So thanks! I can hear my score playing. =) The bad news is that there's a constant extra sound, sounds like a distorted wave that is quite loud, and always present, and I can hear the score quieter, but also badly distorted. Maybe it has something to do with buffer sizes or sample types? If anyone can tell from looking at this where I might be wrong, that would be very helpful.
jack is set to a buffer size of 128 ( this runs no problem when not outputing through the spout ) argc/argv are: -+rtaudio=null -b128 myorc.orc myscore.sco Here's what I have in my engine class:
// setup csound void Engine::initCsound(int argc, char **argv){ cout << "Engine::initCsound()\n"; csound = new Csound(); // TODO: some error handling here, yo int result = csound->PreCompile(); // tell csound not to handle audio i/o itself // the 128 matches the jack buff size and the csound buffsize csound->SetHostImplementedAudioIO(1, 128); result = csound->Compile(argc, argv); // output needs to be scaled csoundScale = csound->Get0dBFS(); // save ref to the csound spout buffer, csoundOuput is a MYFLT* csoundOutput = csound->GetSpout(); } // perform a csound Buffer MYFLT* Engine::performBuffer(){ int result = csound->PerformBuffer(); return csoundOutput; } And here's how it gets used in the jack audio process callback, (the csound orc is mono for simplicity)
int process (jack_nframes_t nframes, void *data){ // cast the void pointer to our engine Engine *engine = (Engine *)data; // grab our output buffer sample_t *out = (sample_t *) jack_port_get_buffer(output_port, nframes); // perform one csound buffer, this copies audio to engine->csoundOuput MYFLT *csoundOutput = engine->performBuffer(); // iterate through the jack buffer copying output from csound for(jack_nframes_t i=0; i < nframes; i++){ out[i] = (sample_t)( csoundOutput[i] * engine->csoundScale ); } // let jack know all is a-ok return 0; } Any advice would be wonderful!
Iain
|
Date | 2011-12-22 07:16 |
From | Iain Duncan |
Subject | [Csnd] Re: having problems using csound spin with jack |
Interestingly, the output is affected by changing ksmps. Both the pitch of the whine and the playback pitch of distorted score change a lot when I move ksmps around. When I set ksmps to 128, the extra whine sound goes away though score output is still distorted. I did verify that the system is capable of playing the thing fine at lower buffer settings than I'm using by trying it out with csound handling audio output. thanks!
Iain On Wed, Dec 21, 2011 at 11:09 PM, Iain Duncan <iainduncanlists@gmail.com> wrote:
|
Date | 2011-12-22 08:23 |
From | Victor Lazzarini |
Subject | Re: [Csnd] Re: having problems using csound spin with jack |
Sounds like your output buffers might not be completely full? Csound buffersizes are counted in samples not in frames (see csound.c). Make sure sizes are matching. On 22 Dec 2011, at 07:16, Iain Duncan wrote:
Dr Victor Lazzarini Senior Lecturer Dept. of Music NUI Maynooth Ireland tel.: +353 1 708 3545 Victor dot Lazzarini AT nuim dot ie |
Date | 2011-12-22 19:08 |
From | Iain Duncan |
Subject | Re: [Csnd] Re: having problems using csound spin with jack |
On Thu, Dec 22, 2011 at 12:23 AM, Victor Lazzarini <Victor.Lazzarini@nuim.ie> wrote:
Thanks Victor. What is the difference between jack frames and samples? I thought they were the same, oops! Do you know if the cast from MYFLT to jack sample_t should be ok?
BTW, I'm enjoying the Audio Programming Book. =) iain |
Date | 2011-12-22 19:24 |
From | Victor Lazzarini |
Subject | Re: [Csnd] Re: having problems using csound spin with jack |
No, what I mean is that csound buffer sizes are counted in samples, not frames. So, if you have a stereo buffer, size 128, it will have 64 frames. Confusingly, spout/spin will be ksmps frames long. But your code is reading from the buffer, so just watch out. This is something that sometimes gets overlooked. Victor On 22 Dec 2011, at 19:08, Iain Duncan wrote: On Thu, Dec 22, 2011 at 12:23 AM, Victor Lazzarini <Victor.Lazzarini@nuim.ie> wrote: Dr Victor Lazzarini Senior Lecturer Dept. of Music NUI Maynooth Ireland tel.: +353 1 708 3545 Victor dot Lazzarini AT nuim dot ie |
Date | 2011-12-22 19:33 |
From | Iain Duncan |
Subject | Re: [Csnd] Re: having problems using csound spin with jack |
On Thu, Dec 22, 2011 at 11:24 AM, Victor Lazzarini <Victor.Lazzarini@nuim.ie> wrote:
Um, I'm feeling dense but I just got totally lost reading the above. I have (for now) only one channel of output in both jack and csound just to get things working first. When I use the stk in my host, I copy the output of one stk tick ( 1 sample ) into one jack frame and all sounds ok. With csound, I thought that calling csound->performBuffer meant that spout was now filled with one csound buffers worth of samples, 128 samples if -b128.
Are you saying that after calling csound->PerformBuffer, I should read ksmp * -b samples from spout? is -b a buffer of ksmps? Or if not, can anyone tell me how much of what I need to copy? I'm not attached to using performBuffer, I could use performKsmp too if that would be easier. How many samples are in the spout after calling peformKsmp?
thanks! Iain |
Date | 2011-12-22 19:54 |
From | Victor Lazzarini |
Subject | Re: [Csnd] Re: having problems using csound spin with jack |
If you are using performBuffer(), you should access the output buffer (csoundGetOutputBuffer()); if you are using performKsmps(), you should access spout. The sizes are as follows: spout -> ksmps*nchnls samples OutputBuffer: csoundGetOutputBufferSize() samples On 22 Dec 2011, at 19:33, Iain Duncan wrote: On Thu, Dec 22, 2011 at 11:24 AM, Victor Lazzarini <Victor.Lazzarini@nuim.ie> wrote: Dr Victor Lazzarini Senior Lecturer Dept. of Music NUI Maynooth Ireland tel.: +353 1 708 3545 Victor dot Lazzarini AT nuim dot ie |
Date | 2011-12-22 20:04 |
From | Victor Lazzarini |
Subject | Re: [Csnd] having problems using csound spin with jack |
I am just reading your code and I see you have this: csoundOutput = csound->GetSpout(); which should be csoundOutput = csound->GetOutputBuffer(); === Otherwise // perform a csound Buffer MYFLT* Engine::performBuffer(){ int result = csound->PerformBuffer(); return csoundOutput; } needs to be MYFLT* Engine::performBuffer(){ int result = csound->PerformKsmps(); return csoundOutput; } === It's useful to keep 'result' though, as it will be non-zero if you reached the end of performance. On 22 Dec 2011, at 07:09, Iain Duncan wrote: > First, thanks to Steven and Rory's help, I've gotten tantalizingly close. So thanks! I can hear my score playing. =) > > The bad news is that there's a constant extra sound, sounds like a distorted wave that is quite loud, and always present, and I can hear the score quieter, but also badly distorted. Maybe it has something to do with buffer sizes or sample types? If anyone can tell from looking at this where I might be wrong, that would be very helpful. > > jack is set to a buffer size of 128 ( this runs no problem when not outputing through the spout ) > argc/argv are: -+rtaudio=null -b128 myorc.orc myscore.sco > > Here's what I have in my engine class: > // setup csound > void Engine::initCsound(int argc, char **argv){ > cout << "Engine::initCsound()\n"; > csound = new Csound(); > // TODO: some error handling here, yo > int result = csound->PreCompile(); > // tell csound not to handle audio i/o itself > // the 128 matches the jack buff size and the csound buffsize > csound->SetHostImplementedAudioIO(1, 128); > result = csound->Compile(argc, argv); > // output needs to be scaled > csoundScale = csound->Get0dBFS(); > // save ref to the csound spout buffer, csoundOuput is a MYFLT* > csoundOutput = csound->GetSpout(); > } > // perform a csound Buffer > MYFLT* Engine::performBuffer(){ > int result = csound->PerformBuffer(); > return csoundOutput; > } > > And here's how it gets used in the jack audio process callback, (the csound orc is mono for simplicity) > > int process (jack_nframes_t nframes, void *data){ > // cast the void pointer to our engine > Engine *engine = (Engine *)data; > // grab our output buffer > sample_t *out = (sample_t *) jack_port_get_buffer(output_port, nframes); > // perform one csound buffer, this copies audio to engine->csoundOuput > MYFLT *csoundOutput = engine->performBuffer(); > // iterate through the jack buffer copying output from csound > for(jack_nframes_t i=0; i < nframes; i++){ > out[i] = (sample_t)( csoundOutput[i] * engine->csoundScale ); > } > // let jack know all is a-ok > return 0; > } > > > Any advice would be wonderful! > Iain Dr Victor Lazzarini Senior Lecturer Dept. of Music NUI Maynooth Ireland tel.: +353 1 708 3545 Victor dot Lazzarini AT nuim dot ie Send bugs reports to the Sourceforge bug tracker https://sourceforge.net/tracker/?group_id=81968&atid=564599 Discussions of bugs and features can be posted here To unsubscribe, send email sympa@lists.bath.ac.uk with body "unsubscribe csound" |
Date | 2011-12-22 20:14 |
From | Iain Duncan |
Subject | Re: [Csnd] having problems using csound spin with jack |
Thanks Victor, maybe when I get this figured out I could add to the comments in csound.h, I got confused reading them. Could you tell me if I'm correct that: - spout, returned by csoundGetSpout, gets filled with ( ksmp * number of channels ) samples after a call to csoundPerformKsmp. Or is it (ksmp * number of channels * -b) samples?
- the output buffer (not the same thing as spout), returned by csoundGetOutputBuffer, gets filled with ( -b * number of channels ) samples after a call to csoundPerformBuffer thanks so much, will try them both out.
Iain On Thu, Dec 22, 2011 at 12:04 PM, Victor Lazzarini <Victor.Lazzarini@nuim.ie> wrote: I am just reading your code and I see you have this: |
Date | 2011-12-22 20:56 |
From | Michael Gogins |
Subject | Re: [Csnd] Re: having problems using csound spin with jack |
In PCM audio ("digital audio") of the sort that soundfiles and Csound deal in, a sample is one scalar number. A sample frame is one sample for every channel of audio. So a mono frame has 1 sample, a stereo frame has 2 samples, a quad frame as 4 samples, and so on. When people say "sampling rate" what they actually mean is "audio sample frames per second" not "samples per second". Buffers can be sized in raw bytes, samples, or frames. In addition, there is a hierarchy of buffers of different sizes. The lowest level buffer is the one that the synthesizer actually accumulates audio in before sending that audio out. In Csound, this is the spout buffer. Then, Csound has an intermediate buffer, and the audio output device driver has its own buffer; the Csound intermediate buffer is some integral fraction of the audio output device driver buffer. Any or all of these buffers may be sized in raw bytes, samples, or frames. E.g. spout for 2 channels at 100 ksmps for floating point samples would be 100 frames (that is what ksmps are, frames), 200 samples, 800 bytes; with double-precision samples, that would be 100 frames, 200 samples, 1600 bytes. Csound works with Jack both through my Jacko opcodes and through the command-line jack option. For the Jacko opcodes, the following conditions must obtain, these may or may not shed light on the command-line Jack driver: Csound's sr (actually, frames per second) must be equal to the Jack daemon's frames per second. Csound's ksmps (actually, frames per kperiod) must be equal to the Jack daemon's frames per period. Frames per period must not only (a) be a power of 2, but also (b) go evenly into the frames per second, e.g. 128 frames per period goes into 48000 frames per second 375 times, for a latency or MIDI time granularity of about 2.7 milliseconds (as good as or better than the absolute best human performers). Jack is not as flexible as Csound when it comes to frames per period etc., so first get Jack working to your satisfaction without Csound, then configure Csound to work with Jack's frames per second and frames per period. Hope this helps, Mike On Thu, Dec 22, 2011 at 2:54 PM, Victor Lazzarini |
Date | 2011-12-22 21:02 |
From | Iain Duncan |
Subject | Re: [Csnd] having problems using csound spin with jack |
Success with csoundPerformBuffer! Thanks Victor. =) For anyone checking this out from the future, this is what is working currently: - jack is running with frames/period = 64, periods/buffer=2
- the csound options to compileCsound do not include and -b - csound->SetHostImplementedAudioIO(1, 64); is being called with the same buffer size as jack - the csound orc is output values noramlized from 0 to 1
- the samples from the csound Output Buffer are being multiplied by: csoundScale = csound->Get0dBFS(); the jack audio callback looks like this (note, both jack and csound are 1 channel )
// our audio callback, called once per output buffer samples int process (jack_nframes_t nframes, void *data){ // cast the void pointer to our engine Engine *engine = (Engine *)data; // grab our output buffer sample_t *out = (sample_t *) jack_port_get_buffer(output_port, nframes); // perform one csound buffer, this copies audio to engine->csoundOuputBuf MYFLT *csoundOutputBuf = engine->performBuffer(); // iterate through the jack buffer copying output from csound output buf for(jack_nframes_t i=0; i < nframes; i++){ out[i] = (sample_t)( csoundOutputBuf[i] * engine->csoundScale ); } // let jack know all is a-ok return 0; } On Thu, Dec 22, 2011 at 12:14 PM, Iain Duncan <iainduncanlists@gmail.com> wrote:
|
Date | 2011-12-22 21:06 |
From | Iain Duncan |
Subject | Re: [Csnd] Re: having problems using csound spin with jack |
Very helpful, may I incorporate that into my write up of this process? I could not find an example tutorial online of using the csoundAPI with jack this way, so I'd like to write one.
Thanks Iain On Thu, Dec 22, 2011 at 12:56 PM, Michael Gogins <michael.gogins@gmail.com> wrote: In PCM audio ("digital audio") of the sort that soundfiles and Csound |
Date | 2011-12-22 21:07 |
From | Victor Lazzarini |
Subject | Re: [Csnd] having problems using csound spin with jack |
On 22 Dec 2011, at 20:14, Iain Duncan wrote:
yes, and the size of spout is ksmps * nchnls
Yes, the buffer gets filled after that call and it is not the same as spout. This is the csound output buffer. . -b is the size of this buffer. You can retrieve this size with csoundGetOutputBufferSize().
Dr Victor Lazzarini Senior Lecturer Dept. of Music NUI Maynooth Ireland tel.: +353 1 708 3545 Victor dot Lazzarini AT nuim dot ie |
Date | 2011-12-22 21:11 |
From | Iain Duncan |
Subject | Re: [Csnd] Re: having problems using csound spin with jack |
Couple more questions that may be useful in my write up, if you have time to answer them. If not, I can prob figure them out by experimenting I guess if I want to use csound as one uses the STK, I would set ksmps to 1, and use spout with csoundPerformKsmp?
I'm also guessing that the best way to do ksmp processing is probably to make sure that ksmp is an even divisor of the jack buffer size, so that in the jack audio callback one can call csoundPerformKsmp a certain number of times per buffer.
This prob also means that the jack frames/period should always be at least ksmp in size? ideally a multiple of ksmp? thanks Iain
|
Date | 2011-12-22 21:26 |
From | Victor Lazzarini |
Subject | Re: [Csnd] Re: having problems using csound spin with jack |
I would not set ksmps=1, it's not effiicient at all. I am not sure what STK does, but as far as Csound is concerned, there is a big performance penalty. Ideally you should leave setting ksmps out of the picture and use the output buffer. Depending on how you write your code, if you are using performKsmps() you might need or might not need to have an even division. It's up to how you do it. If you are copying sample by sample from spout to jack buffer, then when you fill the buffer, you call performKsmps. This way there is no need to be an even division. Victor On 22 Dec 2011, at 21:11, Iain Duncan wrote: > Couple more questions that may be useful in my write up, if you have time to answer them. If not, I can prob figure them out by experimenting > > I guess if I want to use csound as one uses the STK, I would set ksmps to 1, and use spout with csoundPerformKsmp? > > I'm also guessing that the best way to do ksmp processing is probably to make sure that ksmp is an even divisor of the jack buffer size, so that in the jack audio callback one > can call csoundPerformKsmp a certain number of times per buffer. > > This prob also means that the jack frames/period should always be at least ksmp in size? ideally a multiple of ksmp? > > thanks > Iain Dr Victor Lazzarini Senior Lecturer Dept. of Music NUI Maynooth Ireland tel.: +353 1 708 3545 Victor dot Lazzarini AT nuim dot ie Send bugs reports to the Sourceforge bug tracker https://sourceforge.net/tracker/?group_id=81968&atid=564599 Discussions of bugs and features can be posted here To unsubscribe, send email sympa@lists.bath.ac.uk with body "unsubscribe csound" |
Date | 2011-12-22 21:38 |
From | Iain Duncan |
Subject | Re: [Csnd] Re: having problems using csound spin with jack |
On Thu, Dec 22, 2011 at 1:26 PM, Victor Lazzarini <Victor.Lazzarini@nuim.ie> wrote:I would not set ksmps=1, it's not effiicient at all. I am not sure what STK does, but as far as Csound is concerned, there is a big performance penalty. right, more of a just a test to make to eliminate variables. It works, though I discovered that when I use spout, I should not multiple by the value of csoundGet0Db, what's the deal with the difference in amplitude between the buffer and spout?
Off to play (gasp!) acoustic jazz, but thanks so much for the help today Michael and Victor. Hopefully I can produce a document that's a useful reference. Thanks
iain |
Date | 2011-12-22 22:02 |
From | Victor Lazzarini |
Subject | Re: [Csnd] Re: having problems using csound spin with jack |
In general, if you need normalised samples from spout, you should divide them by the value returned by csoundGet0dBFS(). (I can't remember if this is necessary or not for the output buffer) On 22 Dec 2011, at 21:38, Iain Duncan wrote: On Thu, Dec 22, 2011 at 1:26 PM, Victor Lazzarini <Victor.Lazzarini@nuim.ie> wrote:I would not set ksmps=1, it's not effiicient at all. I am not sure what STK does, but as far as Csound is concerned, there is a big performance penalty. Dr Victor Lazzarini Senior Lecturer Dept. of Music NUI Maynooth Ireland tel.: +353 1 708 3545 Victor dot Lazzarini AT nuim dot ie |