mirror of
https://github.com/alsa-project/alsa-utils
synced 2025-01-05 05:46:41 +01:00
Update to speaker-tools 0.0.7
Updated to speaker-tools version 0.0.7.
This commit is contained in:
parent
d0dfabdfcc
commit
ead77a1000
3 changed files with 129 additions and 29 deletions
|
@ -1,4 +1,5 @@
|
||||||
INCLUDES = -I$(top_srcdir)/include
|
SPEAKER_TEST_VERSION = 0.0.7
|
||||||
|
INCLUDES = -I$(top_srcdir)/include -DVERSION=\"$(SPEAKER_TEST_VERSION)\"
|
||||||
bin_PROGRAMS = speaker-test
|
bin_PROGRAMS = speaker-test
|
||||||
speaker_test_SOURCES = speaker-test.c
|
speaker_test_SOURCES = speaker-test.c
|
||||||
man_MANS = speaker-test.1
|
man_MANS = speaker-test.1
|
||||||
|
|
|
@ -4,11 +4,9 @@ make
|
||||||
|
|
||||||
To test: -
|
To test: -
|
||||||
1) Just stereo sound from one stereo jack: -
|
1) Just stereo sound from one stereo jack: -
|
||||||
./speaker-test -Dfront -c2
|
./speaker-test -Dplug:front -c2
|
||||||
2) A 4 speaker setup from two stereo jacks: -
|
2) A 4 speaker setup from two stereo jacks: -
|
||||||
./speaker-test -Dsurround40 -c4
|
./speaker-test -Dplug:surround40 -c4
|
||||||
3) A 5.1 speaker setup from three stereo jacks: -
|
3) A 5.1 speaker setup from three stereo jacks: -
|
||||||
./speaker-test -Dsurround51 -c6
|
./speaker-test -Dplug:surround51 -c6
|
||||||
4) To send a nice low 75Hz tone to the Woofer and then exit without touching any other speakers: -
|
|
||||||
./speaker-test -Dplug:surround51 -c6 -s1 -f75
|
|
||||||
|
|
||||||
|
|
|
@ -23,6 +23,9 @@
|
||||||
* Main program by James Courtier-Dutton (including some source code fragments from the alsa project.)
|
* Main program by James Courtier-Dutton (including some source code fragments from the alsa project.)
|
||||||
* Some cleanup from Daniel Caujolle-Bert <segfault@club-internet.fr>
|
* Some cleanup from Daniel Caujolle-Bert <segfault@club-internet.fr>
|
||||||
*
|
*
|
||||||
|
* Changelog:
|
||||||
|
* 0.0.6 Added support for different sample formats.
|
||||||
|
*
|
||||||
* $Id: speaker_test.c,v 1.00 2003/11/26 19:43:38 jcdutton Exp $
|
* $Id: speaker_test.c,v 1.00 2003/11/26 19:43:38 jcdutton Exp $
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
@ -32,8 +35,10 @@
|
||||||
#include <sched.h>
|
#include <sched.h>
|
||||||
#include <errno.h>
|
#include <errno.h>
|
||||||
#include <getopt.h>
|
#include <getopt.h>
|
||||||
#include "aconfig.h"
|
/* #include "aconfig.h" */
|
||||||
|
|
||||||
|
#define ALSA_PCM_NEW_HW_PARAMS_API
|
||||||
|
#define ALSA_PCM_NEW_SW_PARAMS_API
|
||||||
#include <alsa/asoundlib.h>
|
#include <alsa/asoundlib.h>
|
||||||
#include <sys/time.h>
|
#include <sys/time.h>
|
||||||
#include <math.h>
|
#include <math.h>
|
||||||
|
@ -45,6 +50,7 @@ static unsigned int channels = 1; /* count of channels */
|
||||||
static unsigned int speaker = 0; /* count of channels */
|
static unsigned int speaker = 0; /* count of channels */
|
||||||
static unsigned int buffer_time = 500000; /* ring buffer length in us */
|
static unsigned int buffer_time = 500000; /* ring buffer length in us */
|
||||||
static unsigned int period_time = 100000; /* period time in us */
|
static unsigned int period_time = 100000; /* period time in us */
|
||||||
|
#define PERIODS 4
|
||||||
static double freq = 440; /* sinusoidal wave frequency in Hz */
|
static double freq = 440; /* sinusoidal wave frequency in Hz */
|
||||||
static snd_output_t *output = NULL;
|
static snd_output_t *output = NULL;
|
||||||
static snd_pcm_uframes_t buffer_size;
|
static snd_pcm_uframes_t buffer_size;
|
||||||
|
@ -55,7 +61,33 @@ static const char *channel_name[] = {
|
||||||
"Rear Left" ,
|
"Rear Left" ,
|
||||||
"Rear Right" ,
|
"Rear Right" ,
|
||||||
"Center" ,
|
"Center" ,
|
||||||
"LFE"
|
"LFE",
|
||||||
|
"Side Left",
|
||||||
|
"Side Right",
|
||||||
|
"8",
|
||||||
|
"9",
|
||||||
|
"10",
|
||||||
|
"11",
|
||||||
|
"12",
|
||||||
|
"13",
|
||||||
|
"14",
|
||||||
|
"15",
|
||||||
|
"16"
|
||||||
|
};
|
||||||
|
|
||||||
|
static const int channels4[] = {
|
||||||
|
0,
|
||||||
|
1,
|
||||||
|
3,
|
||||||
|
2
|
||||||
|
};
|
||||||
|
static const int channels6[] = {
|
||||||
|
0,
|
||||||
|
4,
|
||||||
|
1,
|
||||||
|
3,
|
||||||
|
2,
|
||||||
|
5
|
||||||
};
|
};
|
||||||
|
|
||||||
static void generate_sine(signed short *samples, int channel, int count, double *_phase) {
|
static void generate_sine(signed short *samples, int channel, int count, double *_phase) {
|
||||||
|
@ -63,18 +95,46 @@ static void generate_sine(signed short *samples, int channel, int count, double
|
||||||
double max_phase = 1.0 / freq;
|
double max_phase = 1.0 / freq;
|
||||||
double step = 1.0 / (double)rate;
|
double step = 1.0 / (double)rate;
|
||||||
double res;
|
double res;
|
||||||
int chn, ires;
|
int chn;
|
||||||
|
int32_t ires;
|
||||||
|
int8_t *samp8 = (int8_t*) samples;
|
||||||
|
int16_t *samp16 = (int16_t*) samples;
|
||||||
|
int32_t *samp32 = (int32_t*) samples;
|
||||||
|
int sample_size_bits = snd_pcm_format_width(format);
|
||||||
|
|
||||||
while (count-- > 0) {
|
while (count-- > 0) {
|
||||||
res = sin((phase * 2 * M_PI) / max_phase - M_PI) * 32767;
|
//res = sin((phase * 2 * M_PI) / max_phase - M_PI) * 32767;
|
||||||
|
res = (sin((phase * 2 * M_PI) / max_phase - M_PI)) * 0x07ffffff; /* Don't use MAX volume */
|
||||||
|
//if (res > 0) res = 10000;
|
||||||
|
//if (res < 0) res = -10000;
|
||||||
|
|
||||||
/* printf("%e\n",res); */
|
/* printf("%e\n",res); */
|
||||||
ires = res;
|
ires = res;
|
||||||
|
|
||||||
for(chn=0;chn<channels;chn++) {
|
for(chn=0;chn<channels;chn++) {
|
||||||
if (chn==channel)
|
if (sample_size_bits == 8) {
|
||||||
*samples++ = ires;
|
if (chn==channel) {
|
||||||
else
|
*samp8++ = ires >> 24;
|
||||||
*samples++ = 0;
|
//*samp8++ = 0x12;
|
||||||
|
} else {
|
||||||
|
*samp8++ = 0;
|
||||||
|
}
|
||||||
|
} else if (sample_size_bits == 16) {
|
||||||
|
if (chn==channel) {
|
||||||
|
*samp16++ = ires >>16;
|
||||||
|
//*samp16++ = 0x1234;
|
||||||
|
} else {
|
||||||
|
*samp16++ = 0;
|
||||||
|
}
|
||||||
|
} else if (sample_size_bits == 32) {
|
||||||
|
if (chn==channel) {
|
||||||
|
*samp32++ = ires;
|
||||||
|
//*samp32++ = 0xF2345678;
|
||||||
|
//printf("res=%lf, ires=%d 0x%x, samp32=0x%x\n",res,ires, ires, samp32[-1]);
|
||||||
|
} else {
|
||||||
|
*samp32++ = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
phase += step;
|
phase += step;
|
||||||
|
@ -156,13 +216,18 @@ static int set_hwparams(snd_pcm_t *handle, snd_pcm_hw_params_t *params, snd_pcm_
|
||||||
err = snd_pcm_hw_params_get_period_size_max(params, &period_size_max,&dir);
|
err = snd_pcm_hw_params_get_period_size_max(params, &period_size_max,&dir);
|
||||||
printf("Buffer size range from %lu to %lu\n",buffer_size_min, buffer_size_max);
|
printf("Buffer size range from %lu to %lu\n",buffer_size_min, buffer_size_max);
|
||||||
printf("Period size range from %lu to %lu\n",period_size_min, period_size_max);
|
printf("Period size range from %lu to %lu\n",period_size_min, period_size_max);
|
||||||
|
printf("Periods = %d\n", PERIODS);
|
||||||
printf("Buffer time size %lu\n",buffer_time_to_size);
|
printf("Buffer time size %lu\n",buffer_time_to_size);
|
||||||
|
|
||||||
buffer_size = buffer_time_to_size;
|
buffer_size = buffer_time_to_size;
|
||||||
|
//buffer_size=8096;
|
||||||
|
buffer_size=15052;
|
||||||
if (buffer_size_max < buffer_size) buffer_size = buffer_size_max;
|
if (buffer_size_max < buffer_size) buffer_size = buffer_size_max;
|
||||||
if (buffer_size_min > buffer_size) buffer_size = buffer_size_min;
|
if (buffer_size_min > buffer_size) buffer_size = buffer_size_min;
|
||||||
period_size=buffer_size/8;
|
//buffer_size=0x800;
|
||||||
buffer_size = period_size*8;
|
period_size = buffer_size/PERIODS;
|
||||||
|
buffer_size = period_size*PERIODS;
|
||||||
|
//period_size = 510;
|
||||||
printf("To choose buffer_size = %lu\n",buffer_size);
|
printf("To choose buffer_size = %lu\n",buffer_size);
|
||||||
printf("To choose period_size = %lu\n",period_size);
|
printf("To choose period_size = %lu\n",period_size);
|
||||||
dir=0;
|
dir=0;
|
||||||
|
@ -210,8 +275,8 @@ static int set_swparams(snd_pcm_t *handle, snd_pcm_sw_params_t *swparams) {
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* start the transfer when a period is full */
|
/* start the transfer when a buffer is full */
|
||||||
err = snd_pcm_sw_params_set_start_threshold(handle, swparams, period_size);
|
err = snd_pcm_sw_params_set_start_threshold(handle, swparams, buffer_size);
|
||||||
if (err < 0) {
|
if (err < 0) {
|
||||||
printf("Unable to set start threshold mode for playback: %s\n", snd_strerror(err));
|
printf("Unable to set start threshold mode for playback: %s\n", snd_strerror(err));
|
||||||
return err;
|
return err;
|
||||||
|
@ -293,7 +358,7 @@ static int write_loop(snd_pcm_t *handle, int channel, int periods, signed short
|
||||||
|
|
||||||
if (err < 0) {
|
if (err < 0) {
|
||||||
if (xrun_recovery(handle, err) < 0) {
|
if (xrun_recovery(handle, err) < 0) {
|
||||||
printf("Write error: %s\n", snd_strerror(err));
|
printf("Write error: %d,%s\n", err, snd_strerror(err));
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
break; /* skip one period */
|
break; /* skip one period */
|
||||||
|
@ -307,7 +372,9 @@ static int write_loop(snd_pcm_t *handle, int channel, int periods, signed short
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void help(void) {
|
static void help(void)
|
||||||
|
{
|
||||||
|
int k;
|
||||||
|
|
||||||
printf(
|
printf(
|
||||||
"Usage: speaker-test [OPTION]... \n"
|
"Usage: speaker-test [OPTION]... \n"
|
||||||
|
@ -316,11 +383,12 @@ static void help(void) {
|
||||||
"-r,--rate stream rate in Hz\n"
|
"-r,--rate stream rate in Hz\n"
|
||||||
"-c,--channels count of channels in stream\n"
|
"-c,--channels count of channels in stream\n"
|
||||||
"-f,--frequency sine wave frequency in Hz\n"
|
"-f,--frequency sine wave frequency in Hz\n"
|
||||||
|
"-F,--format sample format\n"
|
||||||
"-b,--buffer ring buffer size in us\n"
|
"-b,--buffer ring buffer size in us\n"
|
||||||
"-p,--period period size in us\n"
|
"-p,--period period size in us\n"
|
||||||
"-s,--speaker single speaker test. Values 1=Left or 2=right\n"
|
"-s,--speaker single speaker test. Values 1=Left or 2=right\n"
|
||||||
"\n");
|
"\n");
|
||||||
#if 0
|
#if 1
|
||||||
printf("Recognized sample formats are:");
|
printf("Recognized sample formats are:");
|
||||||
for (k = 0; k < SND_PCM_FORMAT_LAST; ++k) {
|
for (k = 0; k < SND_PCM_FORMAT_LAST; ++k) {
|
||||||
const char *s = snd_pcm_format_name(k);
|
const char *s = snd_pcm_format_name(k);
|
||||||
|
@ -340,12 +408,16 @@ int main(int argc, char *argv[]) {
|
||||||
snd_pcm_sw_params_t *swparams;
|
snd_pcm_sw_params_t *swparams;
|
||||||
signed short *samples;
|
signed short *samples;
|
||||||
int chn;
|
int chn;
|
||||||
|
double time1,time2,time3;
|
||||||
|
struct timeval tv1,tv2;
|
||||||
|
|
||||||
struct option long_option[] = {
|
struct option long_option[] = {
|
||||||
{"help", 0, NULL, 'h'},
|
{"help", 0, NULL, 'h'},
|
||||||
{"device", 1, NULL, 'D'},
|
{"device", 1, NULL, 'D'},
|
||||||
{"rate", 1, NULL, 'r'},
|
{"rate", 1, NULL, 'r'},
|
||||||
{"channels", 1, NULL, 'c'},
|
{"channels", 1, NULL, 'c'},
|
||||||
{"frequency", 1, NULL, 'f'},
|
{"frequency", 1, NULL, 'f'},
|
||||||
|
{"format", 1, NULL, 'F'},
|
||||||
{"buffer", 1, NULL, 'b'},
|
{"buffer", 1, NULL, 'b'},
|
||||||
{"period", 1, NULL, 'p'},
|
{"period", 1, NULL, 'p'},
|
||||||
{"speaker", 1, NULL, 's'},
|
{"speaker", 1, NULL, 's'},
|
||||||
|
@ -361,7 +433,7 @@ int main(int argc, char *argv[]) {
|
||||||
while (1) {
|
while (1) {
|
||||||
int c;
|
int c;
|
||||||
|
|
||||||
if ((c = getopt_long(argc, argv, "hD:r:c:f:b:p:s:", long_option, NULL)) < 0)
|
if ((c = getopt_long(argc, argv, "hD:r:c:f:F:b:p:s:", long_option, NULL)) < 0)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
switch (c) {
|
switch (c) {
|
||||||
|
@ -371,6 +443,9 @@ int main(int argc, char *argv[]) {
|
||||||
case 'D':
|
case 'D':
|
||||||
device = strdup(optarg);
|
device = strdup(optarg);
|
||||||
break;
|
break;
|
||||||
|
case 'F':
|
||||||
|
format = snd_pcm_format_value(optarg);
|
||||||
|
break;
|
||||||
case 'r':
|
case 'r':
|
||||||
rate = atoi(optarg);
|
rate = atoi(optarg);
|
||||||
rate = rate < 4000 ? 4000 : rate;
|
rate = rate < 4000 ? 4000 : rate;
|
||||||
|
@ -426,18 +501,23 @@ int main(int argc, char *argv[]) {
|
||||||
printf("Playback device is %s\n", device);
|
printf("Playback device is %s\n", device);
|
||||||
printf("Stream parameters are %iHz, %s, %i channels\n", rate, snd_pcm_format_name(format), channels);
|
printf("Stream parameters are %iHz, %s, %i channels\n", rate, snd_pcm_format_name(format), channels);
|
||||||
printf("Sine wave rate is %.4fHz\n", freq);
|
printf("Sine wave rate is %.4fHz\n", freq);
|
||||||
|
loop:
|
||||||
if ((err = snd_pcm_open(&handle, device, SND_PCM_STREAM_PLAYBACK, 0)) < 0) {
|
while ((err = snd_pcm_open(&handle, device, SND_PCM_STREAM_PLAYBACK, 0)) < 0) {
|
||||||
printf("Playback open error: %s\n", snd_strerror(err));
|
printf("Playback open error: %d,%s\n", err,snd_strerror(err));
|
||||||
exit(EXIT_FAILURE);
|
sleep(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((err = set_hwparams(handle, hwparams, SND_PCM_ACCESS_RW_INTERLEAVED)) < 0) {
|
if ((err = set_hwparams(handle, hwparams, SND_PCM_ACCESS_RW_INTERLEAVED)) < 0) {
|
||||||
printf("Setting of hwparams failed: %s\n", snd_strerror(err));
|
printf("Setting of hwparams failed: %s\n", snd_strerror(err));
|
||||||
|
snd_pcm_close(handle);
|
||||||
|
goto loop;
|
||||||
exit(EXIT_FAILURE);
|
exit(EXIT_FAILURE);
|
||||||
}
|
}
|
||||||
|
//getchar();
|
||||||
if ((err = set_swparams(handle, swparams)) < 0) {
|
if ((err = set_swparams(handle, swparams)) < 0) {
|
||||||
printf("Setting of swparams failed: %s\n", snd_strerror(err));
|
printf("Setting of swparams failed: %s\n", snd_strerror(err));
|
||||||
|
snd_pcm_close(handle);
|
||||||
|
goto loop;
|
||||||
exit(EXIT_FAILURE);
|
exit(EXIT_FAILURE);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -449,16 +529,37 @@ int main(int argc, char *argv[]) {
|
||||||
if (speaker==0) {
|
if (speaker==0) {
|
||||||
while (1) {
|
while (1) {
|
||||||
|
|
||||||
|
gettimeofday(&tv1, NULL);
|
||||||
for(chn = 0; chn < channels; chn++) {
|
for(chn = 0; chn < channels; chn++) {
|
||||||
printf(" - %s\n", channel_name[chn]);
|
int channel=chn;
|
||||||
|
if (channels == 4) {
|
||||||
|
channel=channels4[chn];
|
||||||
|
}
|
||||||
|
if (channels == 6) {
|
||||||
|
channel=channels6[chn];
|
||||||
|
}
|
||||||
|
printf(" %d - %s\n", channel, channel_name[channel]);
|
||||||
|
|
||||||
err = write_loop(handle, chn, ((rate*5)/period_size), samples);
|
err = write_loop(handle, channel, ((rate*3)/period_size), samples);
|
||||||
|
//err = write_loop(handle, 255, ((rate*3)/period_size), samples);
|
||||||
|
|
||||||
if (err < 0) {
|
if (err < 0) {
|
||||||
printf("Transfer failed: %s\n", snd_strerror(err));
|
printf("Transfer failed: %s\n", snd_strerror(err));
|
||||||
exit(EXIT_FAILURE);
|
free(samples);
|
||||||
|
snd_pcm_close(handle);
|
||||||
|
printf("Pausing\n");
|
||||||
|
goto loop ;
|
||||||
|
//pause();
|
||||||
|
//printf("Done Pausing\n");
|
||||||
|
exit(EXIT_SUCCESS);
|
||||||
|
goto loop ;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
gettimeofday(&tv2, NULL);
|
||||||
|
time1 = (double)tv1.tv_sec + ((double)tv1.tv_usec / 1000000.0);
|
||||||
|
time2 = (double)tv2.tv_sec + ((double)tv2.tv_usec / 1000000.0);
|
||||||
|
time3 = time2 - time1;
|
||||||
|
printf("Time per period = %lf\n", time3 );
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
printf(" - %s\n", channel_name[speaker-1]);
|
printf(" - %s\n", channel_name[speaker-1]);
|
||||||
|
|
Loading…
Reference in a new issue