mario@cs.man.ac.uk (Mario Wolczko) (02/21/91)
Here's a quick patch to the sun-supplied "play" program (available on sparcstations and SLCs, SunOS4.1 or higher). Inspired by an earlier program to set the keyboard LEDs, it uses them as a VU meter... No man page, and not necessarily very clean, but it works (I think). Enjoy! Mario Wolczko ______ Dept. of Computer Science Internet: mario@cs.man.ac.uk /~ ~\ The University uucp: mcsun!ukc!man.cs!mario ( __ ) Manchester M13 9PL JANET: mario@uk.ac.man.cs `-': :`-' U.K. Tel: +44-61-275 6146 (FAX: 6280) ____; ;_____________the mushroom project___________________________________ *** play.c.sun Fri Dec 21 18:30:16 1990 --- play.c Tue Feb 5 18:23:19 1991 *************** *** 20,25 **** --- 20,26 ---- #include <multimedia/libaudio.h> #include <multimedia/audio_device.h> + #include <sundev/kbd.h> #define Error (void) fprintf *************** *** 27,37 **** /* Local variables */ char *prog; char prog_desc[] = "Play an audio file"; ! char prog_opts[] = "Viv:d:?"; /* getopt() flags */ char *Stdin = "stdin"; unsigned char buf[1024 * 64]; /* size should depend on sample_rate */ #define MAX_GAIN (100) /* maximum gain */ --- 28,40 ---- /* Local variables */ char *prog; char prog_desc[] = "Play an audio file"; ! char prog_opts[] = "lViv:d:?"; /* getopt() flags */ char *Stdin = "stdin"; unsigned char buf[1024 * 64]; /* size should depend on sample_rate */ + int kbd; /* fd of kbd */ + unsigned char snd[2]; /* 2 buffer for writes to kbd */ #define MAX_GAIN (100) /* maximum gain */ *************** *** 41,47 **** --- 44,65 ---- */ #define SAMPLE_RATE_THRESHOLD (.01) + /* These define the sample used for averaging to calculate the bar graph size */ + #define LN2_SS 8 + #define SAMPLE_SIZE (1 << LN2_SS) + /* how to convert an average to a bar graph, given only 5 different bar sizes */ + static unsigned char bar[] = { + 0, + LED_CAPS_LOCK, + LED_CAPS_LOCK|LED_COMPOSE, + LED_CAPS_LOCK|LED_COMPOSE, + LED_CAPS_LOCK|LED_COMPOSE|LED_SCROLL_LOCK, + LED_CAPS_LOCK|LED_COMPOSE|LED_SCROLL_LOCK, + LED_CAPS_LOCK|LED_COMPOSE|LED_SCROLL_LOCK|LED_NUM_LOCK, + LED_CAPS_LOCK|LED_COMPOSE|LED_SCROLL_LOCK|LED_NUM_LOCK + }; + unsigned Volume = ~0; /* output volume */ double Savevol; /* saved volume level */ int Verbose = FALSE; /* verbose messages */ *************** *** 52,59 **** Audio_hdr Dev_hdr; /* audio header for device */ char *Ifile; /* current filename */ Audio_hdr File_hdr; /* audio header for file */ - /* Global variables */ extern int getopt(); extern int optind; --- 70,77 ---- Audio_hdr Dev_hdr; /* audio header for device */ char *Ifile; /* current filename */ Audio_hdr File_hdr; /* audio header for file */ + int leds = TRUE; /* use leds as vu meter */ /* Global variables */ extern int getopt(); extern int optind; *************** *** 63,73 **** usage() { Error(stderr, "%s -- usage:\n\t%s ", prog_desc, prog); ! Error(stderr, "\t[-iV] [-v #] [-d dev] [file ...]\nwhere:\n"); Error(stderr, "\t-i\tDon't hang if audio device is busy\n"); Error(stderr, "\t-V\tPrint verbose warning messages\n"); Error(stderr, "\t-v #\tSet output volume (0 - %d)\n", MAX_GAIN); Error(stderr, "\t-d dev\tSpecify audio device (default: /dev/audio)\n"); Error(stderr, "\tfile\tList of files to play\n"); Error(stderr, "\t\tIf no files specified, read stdin\n"); exit(1); --- 81,92 ---- usage() { Error(stderr, "%s -- usage:\n\t%s ", prog_desc, prog); ! Error(stderr, "\t[-iV] [-v #] [-d dev] [-l] [file ...]\nwhere:\n"); Error(stderr, "\t-i\tDon't hang if audio device is busy\n"); Error(stderr, "\t-V\tPrint verbose warning messages\n"); Error(stderr, "\t-v #\tSet output volume (0 - %d)\n", MAX_GAIN); Error(stderr, "\t-d dev\tSpecify audio device (default: /dev/audio)\n"); + Error(stderr, "\t-l\tDon't use the keyboard LEDs\n"); Error(stderr, "\tfile\tList of files to play\n"); Error(stderr, "\t\tIf no files specified, read stdin\n"); exit(1); *************** *** 83,91 **** --- 102,125 ---- (void) audio_set_play_gain(Audio_fd, &Savevol); (void) close(Audio_fd); /* close output */ } + if (leds) { + snd[1]= 0 ; write(kbd, snd, 2); + (void) close(kbd); + } exit(1); } + void set_channel(fd) /* choose output port on env var */ + int fd; + { + extern char *getenv(); + unsigned port= getenv("SST_EARPHONES") ? AUDIO_HEADPHONE : AUDIO_SPEAKER; + + (void) audio_set_play_port(fd, &port); + } + + + /* * Play a list of audio files. */ *************** *** 94,100 **** char **argv; { int i; ! int cnt; int err; int ifd; int stdinseen; --- 128,134 ---- char **argv; { int i; ! int cnt, cum_cnt; int err; int ifd; int stdinseen; *************** *** 101,106 **** --- 135,141 ---- double vol; struct stat st; struct sigvec vec; + register unsigned char *bufp; prog = argv[0]; /* save program initiation name */ *************** *** 123,128 **** --- 158,166 ---- case 'i': Immediate = TRUE; break; + case 'l': + leds = FALSE; + break; case '?': usage(); /*NOTREACHED*/ *************** *** 168,173 **** --- 206,213 ---- exit(1); } + set_channel(Audio_fd); + /* Get the device output encoding configuration */ if (audio_get_play_config(Audio_fd, &Dev_hdr) != AUDIO_SUCCESS) { Error(stderr, "%s: %s is not an audio device\n", *************** *** 188,193 **** --- 228,239 ---- } } + if (leds) { /* open keyboard device */ + kbd= open("/dev/kbd", O_WRONLY); + snd[0]= 016; /* code to set leds */ + if (kbd < 0) leds = FALSE; + } + /* Set up SIGINT handler to flush output */ vec.sv_handler = sigint; vec.sv_mask = 0; *************** *** 254,271 **** /* * At this point, we're all ready to copy the data. */ ! while ((cnt = read(ifd, (char *)buf, sizeof (buf))) >= 0) { ! /* If input EOF, write an eof marker */ ! err = write(Audio_fd, (char *)buf, cnt); ! if (err != cnt) { ! Error(stderr, "%s: output error: ", prog); ! perror(""); ! break; } ! if (cnt == 0) ! break; } if (cnt < 0) { Error(stderr, "%s: error reading ", prog); --- 300,358 ---- /* * At this point, we're all ready to copy the data. + * The outer loop grabs a chunk of data. */ ! for (cum_cnt=0; cnt = read(ifd, (char *)buf, sizeof (buf)), cnt > 0;) { ! /* the inner loop breaks the chunk into ! SAMPLE_SIZEd samples, and averages them */ ! for (bufp= buf; cnt >= SAMPLE_SIZE ; ! bufp += SAMPLE_SIZE, cnt -= SAMPLE_SIZE) { ! register unsigned int total= 0, n; ! /* write to the audio channel */ ! err = write(Audio_fd, (char *)bufp, SAMPLE_SIZE); ! if (err != SAMPLE_SIZE) { ! Error(stderr, "%s: output error: ", prog); ! perror(""); ! break; ! } ! ! if (leds) { ! cum_cnt += SAMPLE_SIZE; ! /* this is the averaging calculation */ ! for (n= 0; n < SAMPLE_SIZE; ++n) { ! unsigned char b=bufp[n]; ! /* full wave rectification */ ! total += b&0x80 ? b^0xFF : b^0x7F; ! } ! ! /* now convert to bargraph pattern */ ! /* seven bits of data -- strip 4 */ ! snd[1]= bar[total >> (LN2_SS + 4)]; ! /* and output to leds */ ! write(kbd, snd, 2); ! ! /* this keeps the leds and audio in sync */ ! cum_cnt += SAMPLE_SIZE; ! if (cum_cnt > 8000) { /* approx 1s */ ! audio_drain(Audio_fd,FALSE); ! cum_cnt= 0; ! } ! } } ! /* remaining chunk: don't average */ ! if (leds) { /* this keeps the leds and audio in sync */ ! cum_cnt += SAMPLE_SIZE; ! if (cum_cnt > 8000) { /* approx 1s */ ! audio_drain(Audio_fd,FALSE); ! cum_cnt= 0; ! } ! snd[1]= 0 ; write(kbd, snd, 2); ! } ! write(Audio_fd, (char *)buf, cnt); ! ! /* If input EOF, write an eof marker */ ! if (cnt > 0) write(Audio_fd, (char *)buf, 0); } if (cnt < 0) { Error(stderr, "%s: error reading ", prog);