Subversion Repositories svnkaklik

Rev

Go to most recent revision | Details | Last modification | View Log

Rev Author Line No. Line
261 kaklik 1
//
2
// sidd.c:  A VLF signal monitor.
3
//
4
// author: Paul Nicholson, paul@abelian.demon.co.uk
5
//
6
 
7
#include <stdlib.h>
8
#include <unistd.h>
9
#include <math.h>
10
#include <stdio.h>
11
#include <sys/stat.h>
12
#include <sys/ioctl.h>
13
#include <sys/time.h>
14
#include <sys/param.h>
15
#include <fcntl.h>
16
#include <errno.h>
17
#include <stdarg.h>
18
#include <ctype.h>
19
#include <string.h>
20
#include <signal.h>
21
#include <time.h>
22
#include <sched.h>
23
#include <linux/soundcard.h>
24
 
25
#include <fftw3.h>
26
 
27
///////////////////////////////////////////////////////////////////////////////
28
//  Tuneable Settings                                                        //
29
///////////////////////////////////////////////////////////////////////////////
30
 
31
#define VERSION "0.93"
32
 
33
//
34
//  Number of soundcard bytes to read at a time.
35
#define NREAD 2048
36
 
37
//
38
//  Max number of bands which can be read from the config file.
39
#define MAXBANDS 20
40
 
41
//
42
//  Name of the configuration file.
43
#define CONFIG_FILE "sidd.conf"
44
 
45
///////////////////////////////////////////////////////////////////////////////
46
//  Globals and fixed definitions                                            // 
47
///////////////////////////////////////////////////////////////////////////////
48
//
49
//  Default values here are over-ridden by the config file.
50
 
51
int mode = 1;                                          //  1 = mono, 2 = stereo
52
int bits = 16;                                    // Sample width, 8 or 16 bits
53
int BINS = 2048;                                    // Number of frequency bins
54
#define FFTWID (2 * BINS)                    // Number of samples in FFT period
55
 
56
int background = 1;                        // Set zero if running in foreground
57
int fdi;                                                   // Input file handle
58
int fdm;                                                   // Mixer file handle
59
int VFLAG = 0;                                    //  Set non-zero by -v option
60
int MFLAG = 0;                                    //  Set non-zero by -m option
61
 
62
int spec_max = 100;       // Issue a spectrum for every spec_max output records
63
int spec_cnt = 0;
64
int sample_rate = 100000;                                 // Samples per second
65
 
66
int chans = 1;
67
int alert_on = 0;
68
 
69
int priority = 0;                       // Set to 1 if high scheduling priority
70
struct sigaction sa;
71
char mailaddr[100];
72
 
73
double los_thresh = 0;                    // Threshold for loss of signal, 0..1
74
int los_timeout = 0;        // Number of seconds before loss of signal declared
75
 
76
double DF;                                   // Frequency resolution of the FFT
77
int bailout_flag = 0;                           // To prevent bailout() looping
78
int grab_cnt = 0;                       // Count of samples into the FFT buffer
79
 
80
// Mixer gain settings requested by config file.
81
int req_lgain = -1;              // Line gain
82
int req_igain = -1;              // Input gain 
83
int req_rgain = -1;              // Record level
84
 
85
//
86
// Various filenames, contents set by config file.
87
//
88
char logfile[100] = "";
89
char device[100] = "/dev/dsp";
90
char mixer[100] = "/dev/mixer";
91
char spectrum_file[100] = "/tmp/sidspec"; 
92
char datadir[100] = ".";
93
 
94
//
95
// Table of frequency bands to monitor
96
//
97
 
98
struct BAND
99
{
100
   char ident[50];
101
 
102
   int start;
103
   int end;
104
}
105
 bands[MAXBANDS];    // Table of bands to be monitored
106
 
107
int nbands = 0;
108
 
109
//
110
//  Independent state variables and buffers for left and right channels
111
//
112
struct CHAN
113
{
114
   char *name;
115
   double *signal_avg;
116
   double *powspec;
117
   double *fft_inbuf;
118
   fftw_complex *fft_data;
119
   fftw_plan ffp;
120
   double peak;
121
   double sum_sq;
122
   int los_state;
123
   time_t los_time;
124
   FILE *fo;
125
   char fname[100];
126
}
127
 left = { "left" }, right = { "right" };
128
 
129
///////////////////////////////////////////////////////////////////////////////
130
//  Various Utility Functions                                                //
131
///////////////////////////////////////////////////////////////////////////////
132
 
133
//
134
//  Issue a message to the log file, if the verbosity level is high enough...
135
//
136
 
137
void report( int level, char *format, ...)
138
{
139
   va_list ap;
140
   void bailout( char *format, ...);
141
   char temp[ 200];
142
 
143
   if( VFLAG < level) return;
144
 
145
   va_start( ap, format);
146
   vsprintf( temp, format, ap);
147
   va_end( ap);
148
 
149
   if( !logfile[0] || !background)
150
      if( background != 2) fprintf( stderr, "%s\n", temp);
151
 
152
   if( logfile[0])
153
   {
154
      time_t now = time( NULL);
155
      struct tm *tm = gmtime( &now);
156
      FILE *flog = NULL;
157
 
158
      if( (flog = fopen( logfile, "a+")) == NULL)
159
         bailout( "cannot open logfile [%s]: %s", logfile, strerror( errno));
160
 
161
      fprintf( flog, "%04d/%02d/%02d %02d:%02d:%02d %s\n", 
162
                tm->tm_year+1900, tm->tm_mon+1, tm->tm_mday,
163
                tm->tm_hour, tm->tm_min, tm->tm_sec, temp);
164
      fclose( flog);
165
   }
166
}
167
 
168
void alert( char *format, ...)
169
{
170
   FILE *f;
171
   va_list( ap);
172
   char cmd[100], temp[100];
173
 
174
   va_start( ap, format);
175
   vsprintf( temp, format, ap);
176
   va_end( ap);
177
 
178
   report( -1, "%s", temp);
179
 
180
   if( !alert_on || !mailaddr[0]) return;
181
 
182
   sprintf( cmd, "mail -s 'sidd alert' '%s'", mailaddr);
183
   if( (f=popen( cmd, "w")) == NULL)
184
   {
185
      report( 0, "cannot exec [%s]: %s", cmd, strerror( errno));
186
      return;
187
   }
188
 
189
   fprintf( f, "sidd: %s\n", temp);
190
   fclose( f);
191
}
192
 
193
//
194
//  We try to exit the program through here, if possible.
195
//  
196
 
197
void bailout( char *format, ...)
198
{
199
   va_list ap;
200
   char temp[ 200];
201
 
202
   if( bailout_flag) exit( 1);
203
   bailout_flag = 1;
204
   va_start( ap, format);
205
   vsprintf( temp, format, ap);
206
   va_end( ap);
207
 
208
   alert( "terminating: %s", temp);
209
   exit( 1);
210
}
211
 
212
//
213
//  Exit with a message if we get any signals.
214
//  
215
 
216
void handle_sigs( int signum)
217
{
218
   bailout( "got signal %d", signum);
219
}
220
 
221
///////////////////////////////////////////////////////////////////////////////
222
//  Soundcard Setup                                                          //
223
///////////////////////////////////////////////////////////////////////////////
224
 
225
//
226
//  Prepare the input stream, setting up the soundcard if the input
227
//  is a character device.
228
//
229
 
230
void setup_input_stream( void)
231
{
232
   struct stat st;
233
 
234
   report( 1, "taking data from [%s]", device);
235
 
236
   if( (fdi = open( device, O_RDONLY)) < 0)
237
      bailout( "cannot open [%s]: %s", strerror( errno));
238
 
239
   if( fstat( fdi, &st) < 0)
240
      bailout( "cannot stat input stream: %s", strerror( errno));
241
 
242
   if( S_ISCHR( st.st_mode)) 
243
   {
244
      int blksize;
245
      int fragreq = 0x7fff000a;
246
      unsigned int format;
247
      unsigned int req_format = AFMT_S16_LE;
248
      if( bits == 8) req_format = AFMT_U8;
249
 
250
      if (ioctl( fdi, SNDCTL_DSP_SETFRAGMENT, &fragreq))
251
         report( 01, "cannot set fragment size");
252
 
253
      if( ioctl( fdi, SNDCTL_DSP_RESET, NULL) < 0)
254
         bailout( "cannot reset input device");
255
 
256
      chans = mode;
257
      if( ioctl( fdi, SNDCTL_DSP_CHANNELS, &chans) < 0)
258
         bailout( "cannot set channels on input device");
259
 
260
      if( ioctl( fdi, SNDCTL_DSP_GETFMTS, &format) < 0)
261
         bailout( "cannot get formats from input device");
262
 
263
      report( 2, "formats available: %08X", format);
264
      if( (format & req_format) == 0)
265
      {
266
         report( 0, "available dsp modes are %08X", format);
267
         bailout( "unable to set %d bit dsp mode", bits);
268
      }
269
 
270
      format = req_format;
271
      if( ioctl( fdi, SNDCTL_DSP_SETFMT, &format) < 0)
272
         bailout( "cannot set dsp format on input device");
273
 
274
      if( ioctl( fdi, SNDCTL_DSP_GETBLKSIZE, &blksize) < 0)
275
         bailout( "cannot get block size from input device");
276
 
277
      report( 2, "dsp block size: %d", blksize);
278
      if( ioctl( fdi, SNDCTL_DSP_CHANNELS, &chans) < 0)
279
         bailout( "cannot get channels from input device");
280
 
281
      report( 1, "requesting rate %d", sample_rate);
282
      if( ioctl( fdi, SNDCTL_DSP_SPEED, &sample_rate) < 0)
283
         bailout( "cannot set sample rate of input device");
284
 
285
      report( 1, "actual rate set: %d samples/sec", sample_rate);
286
      report( 1, "soundcard channels: %d  bits: %d", chans, bits);
287
   }
288
}
289
 
290
///////////////////////////////////////////////////////////////////////////////
291
//  Output Functions                                                         //
292
///////////////////////////////////////////////////////////////////////////////
293
 
294
void maybe_output_spectrum( void)
295
{
296
   FILE *f;
297
   int i;
298
 
299
   if( ++spec_cnt < spec_max) return;  // Wait for spec_max records
300
   spec_cnt = 0;
301
 
302
   if( !spectrum_file[0]) return;     // Spectrum file not wanted.
303
 
304
   if( (f=fopen( spectrum_file, "w+")) == NULL)
305
      bailout( "cannot open spectrum file %s, %s", strerror( errno));
306
 
307
   if( mode == 1)
308
      for( i=0; i<BINS; i++) fprintf( f, "%.5e %.5e\n", 
309
             (i+0.5) * DF, left.signal_avg[i]/spec_max);
310
   else
311
      for( i=0; i<BINS; i++) fprintf( f, "%.5e %.5e %.5e\n", 
312
             (i+0.5) * DF, left.signal_avg[i]/spec_max,
313
                          right.signal_avg[i]/spec_max);
314
   fclose( f);
315
 
316
   for( i=0; i<BINS; i++) left.signal_avg[i] = 0;
317
   if( mode == 2) for( i=0; i<BINS; i++) right.signal_avg[i] = 0;
318
}
319
 
320
void output_record( struct CHAN *c, char *prefix, double fsecs)
321
{
322
   int i, j;
323
   char test[100];
324
 
325
   if( mode == 1)
326
      sprintf( test, "%s.dat", prefix); 
327
   else
328
      sprintf( test, "%s.%s.dat", prefix, c->name); 
329
 
330
   if( !c->fo || strcmp( test, c->fname))
331
   {
332
      if( c->fo) fclose( c->fo);
333
      strcpy( c->fname, test);
334
      report( 0, "using output file [%s]", c->fname);
335
      if( (c->fo=fopen( c->fname, "a+")) == NULL)
336
         bailout( "cannot open [%s], %s", c->fname, strerror( errno));
337
   }
338
 
339
   fprintf( c->fo, "%.3f %.3f %.3f", fsecs, c->peak, sqrt( c->sum_sq/FFTWID));
340
 
341
   for( i=0; i<nbands; i++)
342
   {
343
      double e = 0;
344
      int n1 = bands[i].start/DF;
345
      int n2 = bands[i].end/DF;
346
      for( j=n1; j<= n2; j++) e += c->powspec[j];
347
      e /= n2 - n1 + 1;
348
      fprintf( c->fo, " %.2e", e);
349
   }
350
   fprintf( c->fo, "\n");
351
   fflush( c->fo);
352
 
353
   c->peak = c->sum_sq = 0;
354
}
355
 
356
void output_records( void)
357
{
358
   struct timeval tv;
359
   struct tm *tm;
360
   double fsecs;
361
   time_t ud;
362
   char prefix[100];
363
 
364
   gettimeofday( &tv, NULL);
365
   fsecs = tv.tv_sec + 1e-6 * tv.tv_usec;
366
   ud = tv.tv_sec - tv.tv_sec % 86400;
367
   tm = gmtime( &ud);
368
   sprintf( prefix, "%s/%02d%02d%02d", datadir,
369
                  tm->tm_year - 100, tm->tm_mon+1, tm->tm_mday);
370
 
371
   output_record( &left, prefix, fsecs);
372
   if( mode == 2) output_record( &right, prefix, fsecs);
373
}
374
 
375
void check_los( struct CHAN *c)
376
{
377
   if( !c->los_state)
378
   {
379
      if( !c->los_time && c->peak < los_thresh) time( &c->los_time);
380
      if( c->los_time && c->peak > los_thresh) c->los_time = 0;
381
      if( c->los_time && c->los_time + los_timeout < time( NULL))
382
      {
383
         c->los_state = 1;
384
         c->los_time = 0;
385
         if( mode == 1) alert( "loss of signal");
386
         else alert( "loss of signal on %s", c->name);
387
      }
388
   }
389
   else
390
   {
391
      if( !c->los_time && c->peak > los_thresh) time( &c->los_time);
392
      if( c->los_time && c->peak < los_thresh) c->los_time = 0;
393
      if( c->los_time && c->los_time + los_timeout < time( NULL))
394
      {
395
         c->los_state = 0;
396
         c->los_time = 0;
397
         if( mode == 1) alert( "signal restored");
398
         else alert( "signal restored on %s", c->name);
399
      }
400
   } 
401
}
402
 
403
///////////////////////////////////////////////////////////////////////////////
404
//  Signal Processing                                                        //
405
///////////////////////////////////////////////////////////////////////////////
406
 
407
void process_fft( struct CHAN *c)
408
{
409
   int i;
410
 
411
   //
412
   //  Do the FFT.  First time through, initialise the fft plan.
413
   //
414
 
415
   if( !c->ffp)
416
      c->ffp = fftw_plan_dft_r2c_1d( FFTWID, c->fft_inbuf, c->fft_data,
417
                           FFTW_ESTIMATE | FFTW_DESTROY_INPUT);
418
 
419
   fftw_execute( c->ffp);
420
 
421
   //
422
   //  Obtain squared amplitude of each bin.
423
   //
424
 
425
   c->powspec[ 0] = 0.0;  // Zero the DC component
426
   for( i=1; i<BINS; i++)
427
   {
428
      double t1 = c->fft_data[ i][0];  
429
      double t2 = c->fft_data[ i][1]; 
430
      c->powspec[ i] = t1*t1 + t2*t2;
431
   }
432
 
433
   //
434
   //  Accumulate average signal levels in each bin.  signal_avg is used
435
   //  only for the spectrum file output.
436
   //
437
 
438
   for( i=0; i<BINS; i++) c->signal_avg[ i] += c->powspec[i];
439
   check_los( c);
440
}
441
 
442
void insert_sample( struct CHAN *c, double f)
443
{
444
   c->sum_sq += f * f;
445
   if( f > c->peak) c->peak = f;
446
   if( f < -c->peak) c->peak = -f;
447
 
448
   c->fft_inbuf[ grab_cnt] = f * sin( grab_cnt/(double) FFTWID * M_PI);
449
}
450
 
451
void maybe_do_fft( void)
452
{
453
   if( ++grab_cnt < FFTWID) return;
454
   grab_cnt = 0;
455
 
456
   process_fft( &left);
457
   if( mode == 2) process_fft( &right);
458
 
459
   output_records();
460
   maybe_output_spectrum(); 
461
}
462
 
463
//
464
// Main signal processing loop.  Never returns.
465
//
466
 
467
void process_signal( void)
468
{
469
   unsigned char buff[ NREAD];
470
 
471
   while( 1) 
472
   {
473
      int i, q;
474
 
475
      if( (q=read( fdi, buff, NREAD)) <= 0) 
476
      {
477
         if( !q || errno == ENOENT || errno == 0) 
478
         {  
479
            sched_yield();
480
            usleep( 50000); 
481
            continue;
482
         }
483
 
484
         report( 0, "input file: read error, count=%d errno=%d", q, errno);
485
         exit( 1);
486
      }
487
 
488
      //  Unpack the input buffer into signed 16 bit words.
489
      //  then scale to -1..+1 for further processing.
490
      //  We use 'chans' to decide if the soundcard is giving stereo or
491
      //  mono samples, rather than 'mode', because some cards will refuse
492
      //  to do mono.  
493
      if( bits == 16)
494
      {
495
         if( chans == 1)
496
         {
497
            for( i=0; i<q; i += 2)
498
            {
499
               int fh = *(short *)(buff + i);
500
 
501
               insert_sample( &left, fh/32768.0);
502
               maybe_do_fft();
503
            }
504
         }
505
         else  // chans must be 2
506
         {
507
            if( mode == 1)
508
               for( i=0; i<q; i += 4)
509
               {
510
                  int fh = *(short *)(buff + i);
511
                  insert_sample( &left, fh/32768.0);
512
                  maybe_do_fft();
513
               }
514
            else  // mode == 2
515
               for( i=0; i<q; i += 4)
516
               {
517
                  int fh = *(short *)(buff + i);
518
                  insert_sample( &left, fh/32768.0);
519
 
520
                  fh = *(short *)(buff + i + 2);
521
                  insert_sample( &right, fh/32768.0);
522
                  maybe_do_fft();
523
               }
524
         }
525
      }
526
      else   // bits must be 8
527
      {
528
         if( chans == 1)
529
         {
530
            for( i=0; i<q; i++)
531
            {
532
               int fh = ((short)buff[i] - 128)*256;
533
               insert_sample( &left, fh/32768.0);
534
               maybe_do_fft();
535
            }
536
         }
537
         else  // chans must be 2
538
         {
539
            if( mode == 1)
540
               for( i=0; i<q; i += 2)
541
               {
542
                  int fh = ((short)buff[i] - 128)*256;
543
                  insert_sample( &left, fh/32768.0);
544
                  maybe_do_fft();
545
               }
546
            else  // mode == 2
547
               for( i=0; i<q; i += 2)
548
               {
549
                  int fh = ((short)buff[i] - 128)*256;
550
                  insert_sample( &left, fh/32768.0);
551
 
552
                  fh = ((short)buff[i+1] - 128)*256;
553
                  insert_sample( &right, fh/32768.0);
554
                  maybe_do_fft();
555
               }
556
         }
557
      }
558
   }
559
}
560
 
561
///////////////////////////////////////////////////////////////////////////////
562
//  Configuration File Stuff                                                 //
563
///////////////////////////////////////////////////////////////////////////////
564
 
565
void config_band( char *ident, char *start, char *end)
566
{
567
   struct BAND *b = bands + nbands++;
568
 
569
   if( nbands == MAXBANDS) bailout( "too many bands specified in config file");
570
 
571
   strcpy( b->ident, ident);
572
   b->start = atoi( start);
573
   b->end = atoi( end);
574
 
575
   report( 1, "band %s %d %d", b->ident, b->start, b->end);
576
}
577
 
578
void load_config( void)
579
{
580
   int lino = 0, nf;
581
   FILE *f;
582
   char buff[100], *p, *fields[20];
583
 
584
   if( (f=fopen( CONFIG_FILE, "r")) == NULL)
585
      bailout( "no config file found");
586
 
587
   while( fgets( buff, 99, f))
588
   {
589
      lino++;
590
 
591
      if( (p=strchr( buff, '\r')) != NULL) *p = 0;
592
      if( (p=strchr( buff, '\n')) != NULL) *p = 0;
593
      if( (p=strchr( buff, ';')) != NULL) *p = 0;
594
 
595
      p = buff;  nf = 0;
596
      while( 1)
597
      {
598
         while( *p && isspace( *p)) p++;
599
         if( !*p) break;
600
         fields[nf++] = p;
601
         while( *p && !isspace( *p)) p++;
602
         if( *p) *p++ = 0;
603
      }
604
      if( !nf) continue;
605
 
606
      if( nf == 4 && !strcasecmp( fields[0], "band")) 
607
         config_band( fields[1], fields[2], fields[3]);
608
      else
609
      if( nf == 2 && !strcasecmp( fields[0], "logfile"))
610
      {
611
         strcpy( logfile, fields[1]);
612
         report( 1, "logfile %s", logfile);
613
      }
614
      else
615
      if( nf == 3 && !strcasecmp( fields[0], "los"))
616
      {
617
         los_thresh = atof( fields[1]);
618
         los_timeout = atoi( fields[2]);
619
         report( 1, "los threshold %.3f, timeout %d seconds", 
620
                    los_thresh, los_timeout);
621
      }
622
      else
623
      if( nf == 2 && !strcasecmp( fields[0], "device"))
624
         strcpy( device, fields[1]);
625
      else
626
      if( nf == 2 && !strcasecmp( fields[0], "mixer"))
627
         strcpy( mixer, fields[1]);
628
      else
629
      if( nf == 2 && !strcasecmp( fields[0], "mode"))
630
      {
631
         if( !strcasecmp( fields[1], "mono")) mode = 1;
632
         else
633
         if( !strcasecmp( fields[1], "stereo")) mode = 2;
634
         else
635
            bailout( "error in config file, line %d", lino);
636
      }
637
      else
638
      if( nf == 2 && !strcasecmp( fields[0], "bits"))
639
      {
640
         bits = atoi( fields[1]);
641
         if( bits != 8 && bits != 16)
642
            bailout( "can only do 8 or 16 bits, config file line %d", lino);
643
      }
644
      else
645
      if( nf == 3 && !strcasecmp( fields[0], "spectrum"))
646
      {
647
         strcpy( spectrum_file, fields[1]);
648
         spec_max = atoi( fields[2]);
649
      }
650
      else
651
      if( nf == 2 && !strcasecmp( fields[0], "sched")
652
                  && !strcasecmp( fields[1], "high"))
653
      {
654
         priority = 1;
655
      }
656
      else
657
      if( nf == 4 && !strcasecmp( fields[0], "gain"))
658
      {
659
         int left = atoi( fields[2]);
660
         int right = atoi( fields[3]);
661
         int gain = (right << 8) | left;
662
 
663
         if( !strcasecmp( fields[1], "line")) req_lgain = gain;
664
         else
665
         if( !strcasecmp( fields[1], "overall")) req_igain = gain;
666
         else
667
         if( !strcasecmp( fields[1], "record")) req_rgain = gain;
668
         else
669
            bailout( "unknown gain control [%s]", fields[1]);
670
      }
671
      else
672
      if( nf == 2 && !strcasecmp( fields[0], "rate"))
673
         sample_rate = atoi( fields[1]);
674
      else
675
      if( nf == 2 && !strcasecmp( fields[0], "bins"))
676
         BINS = atoi( fields[1]);
677
      else
678
      if( nf == 2 && !strcasecmp( fields[0], "datadir"))
679
      {
680
         struct stat st;
681
         strcpy( datadir, fields[1]);
682
         if( stat( datadir, &st) < 0 || !S_ISDIR( st.st_mode))
683
            bailout( "no data directory, %s", datadir);
684
      }
685
      else
686
         bailout( "error in config file, line %d", lino);
687
   }
688
 
689
   fclose( f);
690
}
691
 
692
///////////////////////////////////////////////////////////////////////////////
693
//  Mixer Stuff                                                              //
694
///////////////////////////////////////////////////////////////////////////////
695
 
696
// Actual mixer values, read by open_mixer()
697
int mixer_recmask;      // Recording device mask
698
int mixer_stereo;       // Stereo device mask
699
int mixer_line;         // Line input gain setting
700
int mixer_igain;        // Overall input gain setting
701
int mixer_reclev;       // Recording level setting
702
int mixer_recsrc;       // Mask indicating which inputs are set to record
703
 
704
void open_mixer( void)
705
{
706
   if( (fdm = open( mixer, O_RDWR)) < 0)
707
      bailout( "cannot open [%s]: %s", mixer, strerror( errno));
708
 
709
   // Determine the available mixer recording gain controls.
710
   // We must at least have a line input.
711
 
712
   if( ioctl( fdm, SOUND_MIXER_READ_RECMASK, &mixer_recmask) < 0)
713
      bailout( "cannot read mixer devmask");
714
 
715
   if( (mixer_recmask & SOUND_MASK_LINE) == 0)
716
      bailout( "mixer has no line device");
717
 
718
   if( ioctl( fdm, SOUND_MIXER_READ_STEREODEVS, &mixer_stereo) < 0)
719
      bailout( "cannot read mixer stereodevs");
720
 
721
   if( ioctl( fdm, SOUND_MIXER_READ_RECSRC, &mixer_recsrc) < 0)
722
      bailout( "cannot read mixer recsrc");
723
 
724
   // Read the line input gain.  
725
   if( ioctl( fdm, SOUND_MIXER_READ_LINE, &mixer_line) < 0)
726
      bailout( "cannot read mixer line");
727
 
728
   // Read overall input gain?  Optional.
729
   if( (mixer_recmask & SOUND_MASK_IGAIN) &&
730
        ioctl( fdm, SOUND_MIXER_READ_IGAIN, &mixer_igain) < 0)
731
           bailout( "cannot read mixer igain");
732
 
733
   // Read overall recording level?  Optional.
734
   if( (mixer_recmask & SOUND_MASK_RECLEV) &&
735
        ioctl( fdm, SOUND_MIXER_READ_RECLEV, &mixer_reclev) < 0)
736
           bailout( "cannot read mixer reclev");
737
}
738
 
739
void report_mixer_settings( void)
740
{
741
   report( 1, "mixer: line input is %s", 
742
     mixer_stereo & SOUND_MASK_LINE ? "stereo" : "mono");
743
 
744
   report( 1, "mixer: line input is %s", 
745
     mixer_recsrc & SOUND_MASK_LINE ? "on" : "off");
746
 
747
   report( 1, "mixer: line input gain: left=%d right=%d", 
748
             mixer_line & 0xff, (mixer_line >> 8) & 0xff);
749
 
750
   // Overall input gain?  Optional.
751
   if( mixer_recmask & SOUND_MASK_IGAIN)
752
   {
753
      report( 1, "mixer: igain: left=%d right=%d", 
754
                 mixer_igain & 0xff, (mixer_igain >> 8) & 0xff);
755
   }
756
   else report( 1, "mixer: igain: n/a");
757
 
758
   // Overall recording level?  Optional.
759
   if( mixer_recmask & SOUND_MASK_RECLEV)
760
   {
761
      report( 1, "mixer: reclev: left=%d right=%d", 
762
                mixer_reclev & 0xff, (mixer_reclev >> 8) & 0xff);
763
   }
764
   else report( 1, "mixer: reclev: n/a");
765
 
766
}
767
 
768
void setup_mixer( void)
769
{
770
   if( req_lgain >= 0)
771
   {
772
      report( 1, "requesting line input gains left=%d right=%d",
773
             req_lgain & 0xff, (req_lgain >> 8) & 0xff);
774
 
775
      if( ioctl( fdm, SOUND_MIXER_WRITE_LINE, &req_lgain) < 0 ||
776
          ioctl( fdm, SOUND_MIXER_READ_LINE, &mixer_line) < 0)
777
         bailout( "error setting mixer line gain");
778
 
779
      report( 1, "line input gains set to: left=%d right=%d", 
780
             mixer_line & 0xff, (mixer_line >> 8) & 0xff);
781
   }
782
 
783
   if( req_igain >= 0 &&
784
       (mixer_recmask & SOUND_MASK_IGAIN))
785
   {
786
      report( 1, "requesting overall input gains left=%d right=%d",
787
             req_igain & 0xff, (req_igain >> 8) & 0xff);
788
 
789
      if( ioctl( fdm, SOUND_MIXER_WRITE_IGAIN, &req_igain) < 0 ||
790
          ioctl( fdm, SOUND_MIXER_READ_IGAIN, &mixer_igain) < 0)
791
         bailout( "error setting mixer overall input gain");
792
 
793
      report( 1, "overall input gains set to: left=%d right=%d", 
794
                 mixer_igain & 0xff, (mixer_igain >> 8) & 0xff);
795
   }
796
 
797
   if( req_rgain >= 0 &&
798
       (mixer_recmask & SOUND_MASK_RECLEV))
799
   {
800
      report( 1, "requesting overall record levels left=%d right=%d",
801
             req_rgain & 0xff, (req_rgain >> 8) & 0xff);
802
 
803
      if( ioctl( fdm, SOUND_MIXER_WRITE_RECLEV, &req_rgain) < 0 ||
804
          ioctl( fdm, SOUND_MIXER_READ_RECLEV, &mixer_reclev) < 0)
805
         bailout( "error setting mixer record level");
806
 
807
      report( 1, "mixer record levels set to: left=%d right=%d", 
808
                 mixer_reclev & 0xff, (mixer_reclev >> 8) & 0xff);
809
   }
810
 
811
   mixer_recsrc = SOUND_MASK_LINE;
812
   if( ioctl( fdm, SOUND_MIXER_WRITE_RECSRC, &mixer_recsrc) < 0)
813
      bailout( "cannot set mixer recsrc to line");
814
}
815
 
816
///////////////////////////////////////////////////////////////////////////////
817
//  Main                                                                     //
818
///////////////////////////////////////////////////////////////////////////////
819
 
820
void make_daemon( void)
821
{
822
   int childpid, fd;
823
 
824
   if( (childpid = fork()) < 0)
825
      bailout( "cannot fork: %s", strerror( errno));
826
   else if( childpid > 0) exit( 0);
827
 
828
   if( setpgrp() == -1) bailout( "cannot setpgrp");
829
 
830
   if( (childpid = fork()) < 0)
831
      bailout( "cannot fork: %s", strerror( errno));
832
   else if( childpid > 0) exit( 0);
833
 
834
   for( fd = 0; fd <NOFILE; fd++) if( fd != fdi) close( fd);
835
   errno = 0;
836
   background = 2;
837
}
838
 
839
void initialise_channel( struct CHAN *c)
840
{
841
   int i;
842
 
843
   c->fft_inbuf = (double *) malloc( BINS * 2 * sizeof( double));
844
   c->fft_data = fftw_malloc( sizeof( fftw_complex) * FFTWID);
845
   c->powspec = (double *) malloc( BINS * sizeof( double));
846
   c->signal_avg = (double *) malloc( BINS * sizeof( double));
847
   for( i=0; i<BINS; i++) c->signal_avg[i] = 0;
848
}
849
 
850
void setup_signal_handling( void)
851
{
852
   sa.sa_handler = handle_sigs;
853
   sigemptyset( &sa.sa_mask);
854
   sa.sa_flags = 0;
855
   sigaction( SIGINT, &sa, NULL);
856
   sigaction( SIGTERM, &sa, NULL);
857
   sigaction( SIGHUP, &sa, NULL);
858
   sigaction( SIGQUIT, &sa, NULL);
859
   sigaction( SIGFPE, &sa, NULL);
860
   sigaction( SIGBUS, &sa, NULL);
861
   sigaction( SIGSEGV, &sa, NULL);
862
}
863
 
864
// Set scheduling priority to the minimum SCHED_FIFO value.
865
void set_scheduling( void)
866
{
867
   struct sched_param pa;
868
   int min = sched_get_priority_min( SCHED_FIFO);
869
 
870
   pa.sched_priority = min;
871
   if( sched_setscheduler( 0, SCHED_FIFO, &pa) < 0)
872
      report( -1, "cannot set scheduling priority: %s", strerror( errno));
873
   else
874
      report( 0, "using SCHED_FIFO priority %d", min);
875
}
876
 
877
int main( int argc, char *argv[])
878
{
879
   while( 1)
880
   {
881
      int c = getopt( argc, argv, "vfm");
882
 
883
      if( c == 'v') VFLAG++;
884
      else
885
      if( c == 'm') MFLAG++;
886
      else
887
      if( c == 'f') background = 0;
888
      else if( c == -1) break;
889
      else bailout( "unknown option [%c]", c);
890
   }
891
 
892
   setup_signal_handling();
893
   load_config();
894
   open_mixer();
895
 
896
   if( MFLAG)
897
   {
898
      VFLAG = 1;
899
      background = 0;
900
      report_mixer_settings();
901
      exit( 0);
902
   }
903
 
904
   setup_mixer();
905
 
906
   if( background && !logfile[0]) 
907
      report( -1, "warning: no logfile specified for daemon");
908
 
909
   setup_input_stream();
910
   DF = (double) sample_rate/(double) FFTWID;
911
 
912
   report( 1, "resolution: bins=%d fftwid=%d df=%f", BINS, FFTWID, DF);
913
   report( 1, "spectrum file: %s", spectrum_file); 
914
 
915
   initialise_channel( &left);
916
   if( mode == 2) initialise_channel( &right);
917
 
918
   if( background) make_daemon();
919
   if( priority) set_scheduling();
920
 
921
   report( 0, "sidd version %s: starting work", VERSION);
922
   alert_on = 1;
923
   process_signal();
924
   return 0;
925
}
926