mirror of
				https://github.com/saitohirga/WSJT-X.git
				synced 2025-11-03 13:30:52 -05:00 
			
		
		
		
	- SNR displayed on decode line - Separate transmit() subroutine - Output sent to all_ft2.txt - Display Rx level (hit L to toggle On/Off) - Auto-sequencing (hit A to toggle On/Off)
		
			
				
	
	
		
			348 lines
		
	
	
		
			9.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			348 lines
		
	
	
		
			9.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
#include <stdio.h>
 | 
						|
#include "portaudio.h"
 | 
						|
#include <string.h>
 | 
						|
#include <time.h>
 | 
						|
 | 
						|
int iaa;
 | 
						|
int icc;
 | 
						|
double total_time=0.0;
 | 
						|
 | 
						|
//  Definition of structure pointing to the audio data
 | 
						|
typedef struct
 | 
						|
{
 | 
						|
  int    *iwrite;
 | 
						|
  int    *itx;
 | 
						|
  int    *TxOK;
 | 
						|
  int    *Transmitting;
 | 
						|
  int    *nwave;
 | 
						|
  int    *nright;
 | 
						|
  int     nring;
 | 
						|
  int     nfs;
 | 
						|
  short  *y1;
 | 
						|
  short  *y2;
 | 
						|
  short  *iwave;
 | 
						|
} paTestData;
 | 
						|
 | 
						|
//  Input callback routine:
 | 
						|
static int
 | 
						|
SoundIn( void *inputBuffer, void *outputBuffer,
 | 
						|
		       unsigned long framesPerBuffer,
 | 
						|
		       const PaStreamCallbackTimeInfo* timeInfo, 
 | 
						|
		       PaStreamCallbackFlags statusFlags,
 | 
						|
		       void *userData )
 | 
						|
{
 | 
						|
  paTestData *data = (paTestData*)userData;
 | 
						|
  short *in = (short*)inputBuffer;
 | 
						|
  unsigned int i;
 | 
						|
  static int ia=0;
 | 
						|
 | 
						|
  if(*data->Transmitting) return 0;
 | 
						|
  
 | 
						|
  if(statusFlags!=0) printf("Status flags %d\n",(int)statusFlags);
 | 
						|
 | 
						|
  if((statusFlags&1) == 0) {
 | 
						|
    //increment buffer pointers only if data available
 | 
						|
    ia=*data->iwrite;
 | 
						|
    if(*data->nright==0) {                 //Use left channel for input
 | 
						|
      for(i=0; i<framesPerBuffer; i++) {
 | 
						|
	data->y1[ia] = (*in++);
 | 
						|
	data->y2[ia] = (*in++);
 | 
						|
	ia++;
 | 
						|
      }
 | 
						|
    } else {                               //Use right channel
 | 
						|
      for(i=0; i<framesPerBuffer; i++) {
 | 
						|
	data->y2[ia] = (*in++);
 | 
						|
	data->y1[ia] = (*in++);
 | 
						|
	ia++;
 | 
						|
      }
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  if(ia >= data->nring) ia=0;          //Wrap buffer pointer if necessary
 | 
						|
  *data->iwrite = ia;                  //Save buffer pointer
 | 
						|
  iaa=ia;
 | 
						|
  total_time += (double)framesPerBuffer/12000.0;
 | 
						|
  //  printf("iwrite:  %d\n",*data->iwrite);
 | 
						|
  return 0;
 | 
						|
}
 | 
						|
 | 
						|
//  Output callback routine:
 | 
						|
static int
 | 
						|
SoundOut( void *inputBuffer, void *outputBuffer,
 | 
						|
		       unsigned long framesPerBuffer,
 | 
						|
		       const PaStreamCallbackTimeInfo* timeInfo, 
 | 
						|
		       PaStreamCallbackFlags statusFlags,
 | 
						|
		       void *userData )
 | 
						|
{
 | 
						|
  paTestData *data = (paTestData*)userData;
 | 
						|
  short *wptr = (short*)outputBuffer;
 | 
						|
  unsigned int i,n;
 | 
						|
  static short int n2;
 | 
						|
  static int ic=0;
 | 
						|
  static int TxOKz=0;
 | 
						|
  static clock_t tstart=-1;
 | 
						|
  static clock_t tend=-1;
 | 
						|
  static int nsent=0;
 | 
						|
 | 
						|
  //  printf("txOK:  %d  %d\n",TxOKz,*data->TxOK);
 | 
						|
 | 
						|
  if(*data->TxOK && (!TxOKz)) ic=0;   //Reset buffer pointer to start Tx
 | 
						|
  *data->Transmitting=*data->TxOK;    //Set the "transmitting" flag
 | 
						|
 | 
						|
  if(*data->TxOK)  {
 | 
						|
    if(!TxOKz) {
 | 
						|
      // Start of a transmission
 | 
						|
      tstart=clock();
 | 
						|
      nsent=0;
 | 
						|
      //      printf("Start Tx\n");
 | 
						|
    }
 | 
						|
    TxOKz=*data->TxOK;
 | 
						|
    for(i=0 ; i < framesPerBuffer; i++ )  {
 | 
						|
      n2=data->iwave[ic];
 | 
						|
      *wptr++ = n2;                   //left
 | 
						|
      *wptr++ = n2;                   //right
 | 
						|
      ic++;
 | 
						|
 | 
						|
      if(ic > *data->nwave) {
 | 
						|
	*data->TxOK = 0;
 | 
						|
	*data->Transmitting = 0;
 | 
						|
	*data->iwrite = 0;            //Reset Rx buffer pointer to 0
 | 
						|
	ic=0;
 | 
						|
	tend=clock();
 | 
						|
	double TxT=((double)(tend-tstart))/CLOCKS_PER_SEC;
 | 
						|
	//	printf("End Tx, TxT = %f  nSent = %d\n",TxT,nsent);
 | 
						|
	break;
 | 
						|
      }
 | 
						|
    }
 | 
						|
    nsent += framesPerBuffer;
 | 
						|
  } else {
 | 
						|
    memset((void*)outputBuffer, 0, 2*sizeof(short)*framesPerBuffer);
 | 
						|
  }
 | 
						|
  *data->itx = icc;                    //Save buffer pointer
 | 
						|
  icc=ic;
 | 
						|
  return 0;
 | 
						|
}
 | 
						|
 | 
						|
/*******************************************************************/
 | 
						|
int ft2audio_(int *ndevin, int *ndevout, int *npabuf, int *nright, 
 | 
						|
	      short y1[], short y2[], int *nring, int *iwrite, 
 | 
						|
	      int *itx, short iwave[], int *nwave, int *nfsample, 
 | 
						|
	      int *TxOK, int *Transmitting, int *ngo)
 | 
						|
 | 
						|
{
 | 
						|
  paTestData data;
 | 
						|
  PaStream *instream, *outstream;
 | 
						|
  PaStreamParameters inputParameters, outputParameters;
 | 
						|
  //  PaStreamInfo *streamInfo;
 | 
						|
 | 
						|
  int nfpb = *npabuf;
 | 
						|
  int nSampleRate = *nfsample;
 | 
						|
  int ndevice_in = *ndevin;
 | 
						|
  int ndevice_out = *ndevout;
 | 
						|
  double dSampleRate = (double) *nfsample;
 | 
						|
  PaError err_init, err_open_in, err_open_out, err_start_in, err_start_out;
 | 
						|
  PaError err = 0;
 | 
						|
 | 
						|
  data.iwrite = iwrite;
 | 
						|
  data.itx = itx;
 | 
						|
  data.TxOK = TxOK;
 | 
						|
  data.Transmitting = Transmitting;
 | 
						|
  data.y1 = y1;
 | 
						|
  data.y2 = y2;
 | 
						|
  data.nring = *nring;
 | 
						|
  data.nright = nright;
 | 
						|
  data.nwave = nwave;
 | 
						|
  data.iwave = iwave;
 | 
						|
  data.nfs = nSampleRate;
 | 
						|
 | 
						|
  err_init = Pa_Initialize();                      // Initialize PortAudio
 | 
						|
 | 
						|
  if(err_init) {
 | 
						|
    printf("Error initializing PortAudio.\n");
 | 
						|
    printf("\tErrortext: %s\n\tNumber: %d\n",Pa_GetErrorText(err_init),
 | 
						|
	   err_init);
 | 
						|
    Pa_Terminate();  // I don't think we need this but...
 | 
						|
    return(-1);
 | 
						|
  }
 | 
						|
 | 
						|
  //  printf("Opening device %d for input, %d for output...\n",
 | 
						|
  //         ndevice_in,ndevice_out);
 | 
						|
 | 
						|
  inputParameters.device = ndevice_in;
 | 
						|
  inputParameters.channelCount = 2;
 | 
						|
  inputParameters.sampleFormat = paInt16;
 | 
						|
  inputParameters.suggestedLatency = 0.2;
 | 
						|
  inputParameters.hostApiSpecificStreamInfo = NULL;
 | 
						|
 | 
						|
// Test if this configuration actually works, so we do not run into an
 | 
						|
// ugly assertion
 | 
						|
  err_open_in = Pa_IsFormatSupported(&inputParameters, NULL, dSampleRate);
 | 
						|
 | 
						|
  if (err_open_in == 0) {
 | 
						|
    err_open_in = Pa_OpenStream(
 | 
						|
		       &instream,              //address of stream
 | 
						|
		       &inputParameters,
 | 
						|
		       NULL,
 | 
						|
		       dSampleRate,            //Sample rate
 | 
						|
		       nfpb,                   //Frames per buffer
 | 
						|
		       paNoFlag,
 | 
						|
		       (PaStreamCallback *)SoundIn,  //Callback routine
 | 
						|
		       (void *)&data);  //address of data structure
 | 
						|
 | 
						|
    if(err_open_in) {   // We should have no error here usually
 | 
						|
      printf("Error opening input audio stream:\n");
 | 
						|
      printf("\tErrortext: %s\n\tNumber: %d\n",Pa_GetErrorText(err_open_in),
 | 
						|
	     err_open_in);
 | 
						|
      err = 1;
 | 
						|
    } else {
 | 
						|
      //      printf("Successfully opened audio input.\n");
 | 
						|
    }
 | 
						|
  } else {
 | 
						|
    printf("Error opening input audio stream.\n");
 | 
						|
    printf("\tErrortext: %s\n\tNumber: %d\n",Pa_GetErrorText(err_open_in),
 | 
						|
	   err_open_in);
 | 
						|
    err = 1;
 | 
						|
  }
 | 
						|
 | 
						|
  outputParameters.device = ndevice_out;
 | 
						|
  outputParameters.channelCount = 2;
 | 
						|
  outputParameters.sampleFormat = paInt16;
 | 
						|
  outputParameters.suggestedLatency = 0.2;
 | 
						|
  outputParameters.hostApiSpecificStreamInfo = NULL;
 | 
						|
 | 
						|
// Test if this configuration actually works, so we do not run into an
 | 
						|
// ugly assertion.
 | 
						|
  err_open_out = Pa_IsFormatSupported(NULL, &outputParameters, dSampleRate);
 | 
						|
 | 
						|
  if (err_open_out == 0) {
 | 
						|
    err_open_out = Pa_OpenStream(
 | 
						|
		       &outstream,             //address of stream
 | 
						|
		       NULL,
 | 
						|
		       &outputParameters,
 | 
						|
		       dSampleRate,            //Sample rate
 | 
						|
		       nfpb,                   //Frames per buffer
 | 
						|
		       paNoFlag,
 | 
						|
		       (PaStreamCallback *)SoundOut,  //Callback routine
 | 
						|
		       (void *)&data);         //address of data structure
 | 
						|
 | 
						|
    if(err_open_out) {     // We should have no error here usually
 | 
						|
      printf("Error opening output audio stream!\n");
 | 
						|
      printf("\tErrortext: %s\n\tNumber: %d\n",Pa_GetErrorText(err_open_out),
 | 
						|
	     err_open_out);
 | 
						|
      err += 2;
 | 
						|
    } else {
 | 
						|
      //      printf("Successfully opened audio output.\n");
 | 
						|
    }
 | 
						|
  } else {
 | 
						|
    printf("Error opening output audio stream.\n");
 | 
						|
    printf("\tErrortext: %s\n\tNumber: %d\n",Pa_GetErrorText(err_open_out),
 | 
						|
	   err_open_out);
 | 
						|
    err += 2;
 | 
						|
  }
 | 
						|
 | 
						|
  // if there was no error in opening both streams start them
 | 
						|
  if (err == 0) {
 | 
						|
    err_start_in = Pa_StartStream(instream);             //Start input stream
 | 
						|
    if(err_start_in) {
 | 
						|
      printf("Error starting input audio stream!\n");
 | 
						|
      printf("\tErrortext: %s\n\tNumber: %d\n",Pa_GetErrorText(err_start_in),
 | 
						|
	     err_start_in);
 | 
						|
      err += 4;
 | 
						|
    }
 | 
						|
 | 
						|
    err_start_out = Pa_StartStream(outstream);          //Start output stream
 | 
						|
    if(err_start_out) {
 | 
						|
      printf("Error starting output audio stream!\n");
 | 
						|
      printf("\tErrortext: %s\n\tNumber: %d\n",Pa_GetErrorText(err_start_out),
 | 
						|
	     err_start_out);
 | 
						|
      err += 8;
 | 
						|
    } 
 | 
						|
  }
 | 
						|
 | 
						|
  if (err == 0) printf("Audio streams running normally.\n******************************************************************\n");
 | 
						|
 | 
						|
  while( Pa_IsStreamActive(instream) && (*ngo != 0) && (err == 0) )  {
 | 
						|
    int ic1=0;
 | 
						|
    int ic2=0;
 | 
						|
    if(_kbhit()) ic1 = _getch();
 | 
						|
    if(_kbhit()) ic2 = _getch();
 | 
						|
    // if(ic1!=0 || ic2!=0) printf("%d   %d   %d\n",iaa,ic1,ic2);
 | 
						|
    update_(&total_time,&ic1,&ic2);
 | 
						|
    Pa_Sleep(100);
 | 
						|
  }
 | 
						|
 | 
						|
  Pa_AbortStream(instream);              // Abort stream
 | 
						|
  Pa_CloseStream(instream);             // Close stream, we're done.
 | 
						|
  Pa_AbortStream(outstream);              // Abort stream
 | 
						|
  Pa_CloseStream(outstream);             // Close stream, we're done.
 | 
						|
 | 
						|
  Pa_Terminate();
 | 
						|
 | 
						|
  return(err);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
int padevsub_(int *idevin, int *idevout)
 | 
						|
{
 | 
						|
  int numdev,ndefin,ndefout;
 | 
						|
  int nchin[101], nchout[101];
 | 
						|
  int      i, devIdx;
 | 
						|
  int      numDevices;
 | 
						|
  const PaDeviceInfo *pdi;
 | 
						|
  PaError  err;
 | 
						|
 | 
						|
  Pa_Initialize();
 | 
						|
  numDevices = Pa_GetDeviceCount();
 | 
						|
  numdev = numDevices;
 | 
						|
 | 
						|
  if( numDevices < 0 )  {
 | 
						|
    err = numDevices;
 | 
						|
    Pa_Terminate();
 | 
						|
    return err;
 | 
						|
  }
 | 
						|
 | 
						|
  if ((devIdx = Pa_GetDefaultInputDevice()) > 0) {
 | 
						|
    ndefin = devIdx;
 | 
						|
  } else {
 | 
						|
    ndefin = 0;
 | 
						|
  }
 | 
						|
 | 
						|
  if ((devIdx = Pa_GetDefaultOutputDevice()) > 0) {
 | 
						|
    ndefout = devIdx;
 | 
						|
  } else {
 | 
						|
    ndefout = 0;
 | 
						|
  }
 | 
						|
 | 
						|
  printf("\nAudio     Input    Output     Device Name\n");
 | 
						|
  printf("Device  Channels  Channels\n");
 | 
						|
  printf("------------------------------------------------------------------\n");
 | 
						|
 | 
						|
  for( i=0; i < numDevices; i++ )  {
 | 
						|
    pdi = Pa_GetDeviceInfo(i);
 | 
						|
//    if(i == Pa_GetDefaultInputDevice()) ndefin = i;
 | 
						|
//    if(i == Pa_GetDefaultOutputDevice()) ndefout = i;
 | 
						|
    nchin[i]=pdi->maxInputChannels;
 | 
						|
    nchout[i]=pdi->maxOutputChannels;
 | 
						|
    printf("  %2d       %2d        %2d       %s\n",i,nchin[i],nchout[i],
 | 
						|
	   pdi->name);
 | 
						|
  }
 | 
						|
 | 
						|
  printf("\nUser requested devices:   Input = %2d   Output = %2d\n",
 | 
						|
  	 *idevin,*idevout);
 | 
						|
  printf("Default devices:          Input = %2d   Output = %2d\n",
 | 
						|
  	 ndefin,ndefout);
 | 
						|
  if((*idevin<0) || (*idevin>=numdev)) *idevin=ndefin;
 | 
						|
  if((*idevout<0) || (*idevout>=numdev)) *idevout=ndefout;
 | 
						|
  if((*idevin==0) && (*idevout==0))  {
 | 
						|
    *idevin=ndefin;
 | 
						|
    *idevout=ndefout;
 | 
						|
  }
 | 
						|
  printf("Will open devices:        Input = %2d   Output = %2d\n",
 | 
						|
  	 *idevin,*idevout);
 | 
						|
 | 
						|
  Pa_Terminate();
 | 
						|
 | 
						|
  return 0;
 | 
						|
}
 | 
						|
 |