I've written a softsynth on kubuntu (well, it's a flatpak) that uses ALSA to render the samples.
On my main pc that goes to a USB mixer it works fine.
On my second pc that goes to a straight audio device the audio is all scratchy and glitchy like the sample buffers aren't sync'd right.
I'm sure this is an issue with my code. But ALSA's help is a little slim :)
On my main pc where alsa sounds great, the device has these properties:
08:15:05.950599 00:00.000786 PcSng Syn::Init - sound output='USB Audio CODEC|USB Audio' device='hw:6,0'
08:15:05.952192 00:00.001593 PcSng PCM handle name = 'hw:6,0'
08:15:05.952275 00:00.000083 PcSng PCM state = PREPARED
08:15:05.952313 00:00.000038 PcSng access type = RW_INTERLEAVED
08:15:05.952348 00:00.000035 PcSng format = 'S16_LE' (Signed 16 bit Little Endian)
08:15:05.952381 00:00.000033 PcSng subformat = 'STD' (Standard)
08:15:05.952413 00:00.000032 PcSng channels = 2
08:15:05.952445 00:00.000032 PcSng rate = 44100 bps
08:15:05.952477 00:00.000032 PcSng period time = 1451 us
08:15:05.952509 00:00.000032 PcSng period size = 64 frames
08:15:05.952540 00:00.000031 PcSng buffer time = 2902 us
08:15:05.952572 00:00.000032 PcSng buffer size = 128 frames
08:15:05.952604 00:00.000032 PcSng periods per buffer = 2 frames
08:15:05.952637 00:00.000033 PcSng exact rate = 44100/1 bps
08:15:05.952668 00:00.000031 PcSng significant bits = 16
08:15:05.952700 00:00.000032 PcSng is batch = 1
08:15:05.952731 00:00.000031 PcSng is block transfer = 1
08:15:05.952763 00:00.000032 PcSng is double = 0
08:15:05.952795 00:00.000032 PcSng is half duplex = 0
08:15:05.952826 00:00.000031 PcSng is joint duplex = 0
08:15:05.952857 00:00.000031 PcSng can overrange = 0
08:15:05.952925 00:00.000068 PcSng can mmap = 1
08:15:05.953041 00:00.000116 PcSng can pause = 1
08:15:05.953060 00:00.000019 PcSng can resume = 0
08:15:05.953077 00:00.000017 PcSng can sync start = 0
08:15:05.953119 00:00.000042 PcSng    nFr=64 frq=44100
On my second pc where it's scratchy/glitchy the device has these properties
07:30:45.583342 00:00.000356 PcSng Syn::Init - sound output='HDA Intel PCH|ALC256 Analog' device='hw:6,0'
07:30:45.624918 00:00.041576 PcSng PCM handle name = 'hw:6,0'
07:30:45.624946 00:00.000028 PcSng PCM state = PREPARED
07:30:45.624956 00:00.000010 PcSng access type = RW_INTERLEAVED
07:30:45.624966 00:00.000010 PcSng format = 'S16_LE' (Signed 16 bit Little Endian)
07:30:45.624975 00:00.000009 PcSng subformat = 'STD' (Standard)
07:30:45.624984 00:00.000009 PcSng channels = 2
07:30:45.624993 00:00.000009 PcSng rate = 44100 bps
07:30:45.625002 00:00.000009 PcSng period time = 1451 us
07:30:45.625010 00:00.000008 PcSng period size = 64 frames
07:30:45.625019 00:00.000009 PcSng buffer time = 2902 us
07:30:45.625027 00:00.000008 PcSng buffer size = 128 frames
07:30:45.625036 00:00.000009 PcSng periods per buffer = 2 frames
07:30:45.625045 00:00.000009 PcSng exact rate = 44100/1 bps
07:30:45.625054 00:00.000009 PcSng significant bits = 16
07:30:45.625063 00:00.000009 PcSng is batch = 0
07:30:45.625071 00:00.000008 PcSng is block transfer = 1
07:30:45.625080 00:00.000009 PcSng is double = 0
07:30:45.625089 00:00.000009 PcSng is half duplex = 0
07:30:45.625098 00:00.000009 PcSng is joint duplex = 0
07:30:45.625106 00:00.000008 PcSng can overrange = 0
07:30:45.625115 00:00.000009 PcSng can mmap = 1
07:30:45.625124 00:00.000009 PcSng can pause = 1
07:30:45.625132 00:00.000008 PcSng can resume = 0
07:30:45.625141 00:00.000009 PcSng can sync start = 1
07:30:45.625245 00:00.000104 PcSng    nFr=64 frq=44100
So the difference is ok:
08:15:05.952700 00:00.000032 PcSng is batch = 1
08:15:05.953077 00:00.000017 PcSng can sync start = 0
lame:
07:30:45.625063 00:00.000009 PcSng is batch = 0
07:30:45.625141 00:00.000009 PcSng can sync start = 1
Does this give any help on what my soft synth is doing wrong?
I use 2 period buffers and my synth flips between generating samples into one then the next. And when I dump the audio, I'm using code like this: (sbyt2 is signed short int, ubyt4 is unsigned long int, etc) The code behaves and I don't see these DBG things in my debug log...
void SndO::Put (sbyt2 *buf)
{ ubyt4 p;
  int   e;
   for (p = 0;  p < _nFr;) {
      e = snd_pcm_writei (_hnd, (void *)(& buf [p*2]), _nFr - p);
      if (e >= 0)  p += e;
      else                          // oops - fixup stuff
         switch (e) {
            case -EAGAIN:
               if ((e = snd_pcm_wait    (_hnd, 1)) < 0)
{DBG("pcm_wait died - `s",    snd_strerror (e));   return;}
               break;
            case -ESTRPIPE:
               if ((e = snd_pcm_resume  (_hnd)) < 0)    // n fall thru
{DBG("pcm_resume died - `s",  snd_strerror (e));   return;}
            case -EPIPE:  case -EBADFD:
               if ((e = snd_pcm_prepare (_hnd)) < 0)
{DBG("pcm_prepare died - `s", snd_strerror (e));   return;}
               break;
            default:
{DBG("pcr_writei died - `s",  snd_strerror (e));   return;}
         }
   }
}
My synth has a busy loop (assuming that this Put function does the sync-ing properly) So it's run () is like this:
void Syn::run ()
// live rendering: send interleaved stereo sbyt2 samples to soundcard
{ ubyt4 sz  = _nFr * sizeof (real);
  ubyte per = 0;
  ubyt2   i;
  sbyt2 (*o)[2];
DBGTH("Syn");   DBG("run bgn");
   while (_run) {
DBG("  syn run top");
      o = & _out [per*_nFr];   per = per ? 0 : 1;     // double bufferin
      MemSet (_mixL, 0, sz);   MemSet (_mixR, 0, sz);
      _lok.Grab ();
      for (i = 0;  i < _nVc;  i++)  _vc [i].Mix ();
      for (i = 0;  i < _nFr;  i++) {   // sound card wants interleaved ints
         o [i][0] = r2i (_mixL [i], Dither [0][_dth]);
         o [i][1] = r2i (_mixR [i], Dither [1][_dth]);
         if (++_dth >= MAX_DITHER)  _dth = 0;
      }
      _lok.Toss ();
DBG("  syn run put");
      _sn->Put ((sbyt2 *)o);           // this'll block us on 2nd call and on
DBG("  syn run bot");
   }
DBG("run end");
}
There is very little duration between "syn run top" and "syn run put". So my sample rendering is plenty fast on both pcs. Very little duration between "syn run bot" and "syn run top" as expected - nothin happens there but a jump :) All the duration is between "syn run put" and "syn run bot" where Put is called.
The pc where it runs fine seems to have a more regular duration than on the pc where it doesn't run fine...
Can somebody who knows ALSA help me out here, please?
And if you know of a preferred audio api on linux, please let me know - I'm pretty sheltered and new to linux...
I've heard of pipewire - should I prefer that audio api?
Thanks much.
oops - I open the alsa device with this code:
SndO::SndO (char *dev, ubyt4 inFr, ubyt4 ifrq)
// device to write,  frames in 1 period,  and frequency
// open up our alsa pcm device (audio out)
// always 2 periods of nFr frames - interleaved stereo s16
: _nFr (inFr), _frq (ifrq)             // what we ask fer.  what we get may diff
{ int   e;  // error
  sbyt4 dir  = 0;
  ubyt4 nPer = 2, nFr = inFr, frq = ifrq;
   StrCp (_dev, dev);
   _hnd = nullptr;
   if ((e = snd_pcm_open (& _hnd, _dev, SND_PCM_STREAM_PLAYBACK,
                                        SND_PCM_NONBLOCK))) {
      if (e == -EBUSY)
DBG("snd_pcm_open `s - another app has it - `s", _dev, snd_strerror (e));
      else
DBG("snd_pcm_open `s died - `s",                 _dev, snd_strerror (e));
      _hnd = nullptr;   return;
   }
// SHOISH !!!
  snd_pcm_hw_params_t *hw;
   snd_pcm_hw_params_alloca (& hw);
   snd_pcm_hw_params_any (_hnd, hw);
   if ((e = snd_pcm_hw_params_set_access (_hnd, hw,
                                          SND_PCM_ACCESS_RW_INTERLEAVED)) < 0) {
DBG("pcm_access rw_interleaved died - `s", snd_strerror (e));
      snd_pcm_close (_hnd);   _hnd = nullptr;   return;
   }
   if ((e = snd_pcm_hw_params_set_format (_hnd, hw, SND_PCM_FORMAT_S16)) < 0) {
DBG("pcm_format s16 died - `s", snd_strerror (e));
      snd_pcm_close (_hnd);   _hnd = nullptr;   return;
   }
   if ((e = snd_pcm_hw_params_set_channels (_hnd, hw, 2)) < 0) {
DBG("pcm_channels stereo died - `s", snd_strerror (e));
      snd_pcm_close (_hnd);   _hnd = nullptr;   return;
   }
   if ((e = snd_pcm_hw_params_set_rate_near (_hnd, hw, & frq, 0)) < 0) {
DBG("pcm_rate 44100 died - `s", snd_strerror (e));
      snd_pcm_close (_hnd);   _hnd = nullptr;   return;
   }
   if (_frq != frq)  DBG("pcm_rate wanted `d got `d :/", _frq, frq);
   if ((e = snd_pcm_hw_params_set_period_size_near (_hnd, hw,
                                   (snd_pcm_uframes_t *)(& nFr), & dir)) < 0) {
DBG("pcm_period died - `s", snd_strerror (e));
      snd_pcm_close (_hnd);   _hnd = nullptr;   return;
   }
   if (_nFr != nFr)  DBG("pcm_period wanted `d got `d :/", _nFr, nFr);
   if ((e = snd_pcm_hw_params_set_periods_near (_hnd, hw, & nPer, & dir)) < 0) {
DBG("pcm_periods died - `s", snd_strerror (e));
      snd_pcm_close (_hnd);   _hnd = nullptr;   return;
   }
   if (nPer != 2) {
DBG("pcm_periods is `d not 2 :(", nPer);
      snd_pcm_close (_hnd);   _hnd = nullptr;   return;
   }
   if ((e = snd_pcm_hw_params (_hnd, hw)) < 0) {
DBG("pcm_hw died - `s", snd_strerror (e));
      snd_pcm_close (_hnd);   _hnd = nullptr;   return;
   }
Dump (hw);
  snd_pcm_sw_params_t *sw;
   snd_pcm_sw_params_alloca (& sw);
   snd_pcm_sw_params_current (_hnd, sw);
   if ((e = snd_pcm_sw_params_set_start_threshold (_hnd, sw, _nFr)) < 0) {
DBG("pcm_start_thresh died - `s", snd_strerror (e));
      snd_pcm_close (_hnd);   _hnd = nullptr;   return;
   }
   if ((e = snd_pcm_sw_params_set_avail_min (_hnd, sw, _nFr)) < 0) {
DBG("pcm_avail_min died - `s", snd_strerror (e));
      snd_pcm_close (_hnd);   _hnd = nullptr;   return;
   }
   if ((e = snd_pcm_sw_params (_hnd, sw)) < 0) {
DBG("pcm_sw died - `s", snd_strerror (e));
      snd_pcm_close (_hnd);   _hnd = nullptr;   return;
   }
   if ((e = snd_pcm_nonblock (_hnd, 0)) < 0) {
DBG("pcm_nonblock died - `s", snd_strerror (e));
      snd_pcm_close (_hnd);   _hnd = nullptr;   return;
   }
   if ((e = snd_pcm_prepare (_hnd)) < 0) {
DBG("pcm_prepare died - `s", snd_strerror (e));
      snd_pcm_close (_hnd);   _hnd = nullptr;   return;
   }
}

Syn::run(), you seem to have a logicalk bug, asperis never changed from 0, so thatois never changed from&_out[0]. So, I'm betting on a bug in your overall software.booltype, and casting that type from and to integers is well-defined. So, if I was to write your toggle, I'd just use aboolinstead of yourubyte(and the compiler would potentially be able to run better optimization on bools!), and no matter whether I use aboolor integer type (like yourubyte),!perwould interpret the value astrueorfalse, and then negate it. And:booltimes an integer casts to0(false) or1(true) as per C++ standard.ubyte,sbyt2… types,#include <cstdint>has the standarduint8_t,int16_t, and everybody knows what they are