Csound Csound-dev Csound-tekno Search About

[Csnd] having problems using csound spin with jack

Date2011-12-22 07:09
FromIain 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

Date2011-12-22 07:16
FromIain 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:
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


Date2011-12-22 08:23
FromVictor Lazzarini
SubjectRe: [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.

Victor
On 22 Dec 2011, at 07:16, Iain Duncan wrote:

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:
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




Date2011-12-22 19:08
FromIain Duncan
SubjectRe: [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:
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.

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


Date2011-12-22 19:24
FromVictor Lazzarini
SubjectRe: [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:
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.

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


Dr Victor Lazzarini
Senior Lecturer
Dept. of Music
NUI Maynooth Ireland
tel.: +353 1 708 3545
Victor dot Lazzarini AT nuim dot ie




Date2011-12-22 19:33
FromIain Duncan
SubjectRe: [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:
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.

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


Date2011-12-22 19:54
FromVictor Lazzarini
SubjectRe: [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

Victor

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:
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.

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


Dr Victor Lazzarini
Senior Lecturer
Dept. of Music
NUI Maynooth Ireland
tel.: +353 1 708 3545
Victor dot Lazzarini AT nuim dot ie




Date2011-12-22 20:04
FromVictor Lazzarini
SubjectRe: [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"


Date2011-12-22 20:14
FromIain Duncan
SubjectRe: [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:

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"



Date2011-12-22 20:56
FromMichael Gogins
SubjectRe: [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
 wrote:
> 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
>
> Victor
>
> On 22 Dec 2011, at 19:33, Iain Duncan wrote:
>
> On Thu, Dec 22, 2011 at 11:24 AM, Victor Lazzarini
>  wrote:
>>
>> 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.
>
>
> 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
>
>
> Dr Victor Lazzarini
> Senior Lecturer
> Dept. of Music
> NUI Maynooth Ireland
> tel.: +353 1 708 3545
> Victor dot Lazzarini AT nuim dot ie
>
>
>



-- 
Michael Gogins
Irreducible Productions
http://www.michael-gogins.com
Michael dot Gogins at gmail dot com


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"


Date2011-12-22 21:02
FromIain Duncan
SubjectRe: [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:
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:

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"




Date2011-12-22 21:06
FromIain Duncan
SubjectRe: [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
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
<Victor.Lazzarini@nuim.ie> wrote:
> 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
>
> Victor
>
> 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:
>>
>> 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.
>
>
> 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
>
>
> Dr Victor Lazzarini
> Senior Lecturer
> Dept. of Music
> NUI Maynooth Ireland
> tel.: +353 1 708 3545
> Victor dot Lazzarini AT nuim dot ie
>
>
>



--
Michael Gogins
Irreducible Productions
http://www.michael-gogins.com
Michael dot Gogins at gmail dot com


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"



Date2011-12-22 21:07
FromVictor Lazzarini
SubjectRe: [Csnd] having problems using csound spin with jack

On 22 Dec 2011, at 20:14, Iain Duncan wrote:

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?

yes, and the size of spout is  ksmps * nchnls

- 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

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().



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:

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"



Dr Victor Lazzarini
Senior Lecturer
Dept. of Music
NUI Maynooth Ireland
tel.: +353 1 708 3545
Victor dot Lazzarini AT nuim dot ie




Date2011-12-22 21:11
FromIain Duncan
SubjectRe: [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

Date2011-12-22 21:26
FromVictor Lazzarini
SubjectRe: [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"


Date2011-12-22 21:38
FromIain Duncan
SubjectRe: [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.
Ideally you should leave setting  ksmps out of the picture and use the output buffer.

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


Date2011-12-22 22:02
FromVictor Lazzarini
SubjectRe: [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.
Ideally you should leave setting  ksmps out of the picture and use the output buffer.

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


Dr Victor Lazzarini
Senior Lecturer
Dept. of Music
NUI Maynooth Ireland
tel.: +353 1 708 3545
Victor dot Lazzarini AT nuim dot ie