fixed audio handling for the most part.

aliasing noise and/or buffer underrun noise is present.
This commit is contained in:
gardners 2012-05-11 14:11:58 +09:30
parent 2ea218b567
commit c448eab720

View File

@ -31,12 +31,16 @@
We may need to introduce a low-pass filter to prevent aliasing, assuming that We may need to introduce a low-pass filter to prevent aliasing, assuming that
the microphone and ACD in these phones responds to requencies above 4KHz. the microphone and ACD in these phones responds to requencies above 4KHz.
Added fun with this device is that we must read/write exactly one buffer full
at a time.
*/ */
#define DESIRED_BUFFER_SIZE 256 #define DESIRED_BUFFER_SIZE 256
#define DESIRED_SAMPLE_RATE 32000 #define DESIRED_SAMPLE_RATE 32000
#define RESAMPLE_FACTOR (DESIRED_SAMPLE_RATE/8000) #define RESAMPLE_FACTOR (DESIRED_SAMPLE_RATE/8000)
int resamplingBufferSize=0; int resamplingBufferSize=0;
unsigned char *resamplingBuffer=0; unsigned char *playMarshallBuffer=0;
unsigned char *recordMarshallBuffer=0;
extern int playFd; extern int playFd;
extern int recordFd; extern int recordFd;
@ -259,13 +263,6 @@ int audio_msm_g1_start_play()
{ {
if (playFd>-1) return 0; if (playFd>-1) return 0;
/* Largest possible resampling buffer required for this chipset.
(we resample so that we can use 8KHz audio for transport,
but 32KHz sample rate on the audio hardware so that buffers
correspond to a shorter period of time */
resamplingBufferSize=2*RESAMPLE_FACTOR*8192;
resamplingBuffer=malloc(resamplingBufferSize);
/* Get audio control device */ /* Get audio control device */
int fd = open ("/dev/msm_snd", O_RDWR); int fd = open ("/dev/msm_snd", O_RDWR);
if (fd<0) return -1; if (fd<0) return -1;
@ -324,6 +321,8 @@ int audio_msm_g1_start_play()
float bufferTime=playBufferSize/2*1.0/config.sample_rate; float bufferTime=playBufferSize/2*1.0/config.sample_rate;
WHYF("PLAY buf=%.3fsecs.",bufferTime); WHYF("PLAY buf=%.3fsecs.",bufferTime);
playMarshallBuffer=malloc(playBufferSize);
/* tell hardware to start playing */ /* tell hardware to start playing */
ioctl(playFd,AUDIO_START,0); ioctl(playFd,AUDIO_START,0);
@ -336,7 +335,8 @@ int audio_msm_g1_stop_play()
{ {
WHY("stopping audio play"); WHY("stopping audio play");
if (playFd>-1) close(playFd); if (playFd>-1) close(playFd);
playFd=-1; if (playMarshallBuffer) free(playMarshallBuffer);
playFd=-1; playMarshallBuffer=NULL;
return 0; return 0;
} }
@ -371,6 +371,9 @@ int audio_msm_g1_start_record()
float bufferTime=recordBufferSize/2*1.0/config.sample_rate; float bufferTime=recordBufferSize/2*1.0/config.sample_rate;
WHYF("REC buf=%.3fsecs.",bufferTime); WHYF("REC buf=%.3fsecs.",bufferTime);
if (!recordMarshallBuffer)
recordMarshallBuffer=malloc(recordBufferSize);
fcntl(recordFd,F_SETFL, fcntl(recordFd,F_SETFL,
fcntl(recordFd, F_GETFL, NULL)|O_NONBLOCK); fcntl(recordFd, F_GETFL, NULL)|O_NONBLOCK);
@ -385,6 +388,8 @@ int audio_msm_g1_stop_record()
{ {
WHY("stopping recording"); WHY("stopping recording");
if (recordFd>-1) close(recordFd); if (recordFd>-1) close(recordFd);
if (recordMarshallBuffer) free(recordMarshallBuffer);
recordMarshallBuffer=NULL;
recordFd=-1; recordFd=-1;
return 0; return 0;
} }
@ -417,50 +422,50 @@ int audio_msm_g1_poll_fds(struct pollfd *fds,int slots)
return count; return count;
} }
int recordMarshallBufferOffset=0;
int audio_msm_g1_read(unsigned char *buffer,int maximum_count) int audio_msm_g1_read(unsigned char *buffer,int maximum_count)
{ {
if (recordFd==-1) return 0; if (recordFd==-1) return 0;
if (!resamplingBuffer) return 0; if (!recordMarshallBuffer) return 0;
int maxRawBytes=maximum_count*RESAMPLE_FACTOR; int supplied=0;
if (maxRawBytes>resamplingBufferSize)
maxRawBytes=resamplingBufferSize;
if (maxRawBytes>recordBufferSize)
maxRawBytes=recordBufferSize;
/* read new samples if we don't have any lingering around */
if (!recordMarshallBufferOffset) {
fcntl(recordFd,F_SETFL,fcntl(recordFd, F_GETFL, NULL)|O_NONBLOCK); fcntl(recordFd,F_SETFL,fcntl(recordFd, F_GETFL, NULL)|O_NONBLOCK);
ioctl(recordFd,AUDIO_START,0); ioctl(recordFd,AUDIO_START,0);
WHY("calling read()"); WHY("calling read()");
int b=read(recordFd,&recordMarshallBuffer[0],recordBufferSize);
/* read raw samples */
int b=read(recordFd,&resamplingBuffer[0],maxRawBytes);
if (b<1) if (b<1)
WHYF("read failed: b=%d, err=%s",b,strerror(errno)); WHYF("read failed: b=%d, err=%s",b,strerror(errno));
if (errno==EBADF) recordFd=-1; if (errno==EBADF) recordFd=-1;
WHYF("read %d raw (upsampled) bytes",b); WHYF("read %d raw (upsampled) bytes",b);
recordMarshallBufferOffset=b;
/* downsample to output buffer */
{
int i;
/* copy every RESAMPLE_FACTOR-th sample (each being 16 bits)
to output buffer */
int outpos=0;
for(i=0;i<b;i+=(2*RESAMPLE_FACTOR))
{
buffer[outpos++]=resamplingBuffer[i];
buffer[outpos++]=resamplingBuffer[i+1];
}
b/=RESAMPLE_FACTOR;
}
WHYF("Read %d samples.",b/2);
#warning for debug
WHYF("Echoing %d bytes of 8KHz audio",b);
audio_msm_g1_write(buffer,b);
return b;
} }
/* supply audio from marshalling buffer if it has anything.
Don't forget to downsample first. */
int marshall_offset=0;
while(marshall_offset<recordMarshallBufferOffset
&&supplied<maximum_count) {
buffer[supplied+0]=recordMarshallBuffer[marshall_offset];
buffer[supplied+1]=recordMarshallBuffer[marshall_offset+1];
supplied+=2;
marshall_offset+=2*RESAMPLE_FACTOR;
}
bcopy(&recordMarshallBuffer[marshall_offset],
&recordMarshallBuffer[0],
recordMarshallBufferOffset-marshall_offset);
recordMarshallBufferOffset-=marshall_offset;
/* Else we read exactly one buffer full into the marshalling buffer */
WHYF("Read %d samples.",supplied/2);
return supplied;
}
int playMarshallBufferOffset=0;
int audio_msm_g1_write(unsigned char *data,int bytes) int audio_msm_g1_write(unsigned char *data,int bytes)
{ {
if (playFd==-1) return 0; if (playFd==-1) return 0;
@ -468,20 +473,12 @@ int audio_msm_g1_write(unsigned char *data,int bytes)
WHYF("Writing %d bytes of 8KHz audio",bytes); WHYF("Writing %d bytes of 8KHz audio",bytes);
int maxBytes=bytes; int i,played=0;
if (maxBytes*RESAMPLE_FACTOR>resamplingBufferSize)
maxBytes=resamplingBufferSize/RESAMPLE_FACTOR;
/* upsample ready for play back.
XXX This is a really crude approach, and it could be done much better. */
int i,j,outpos=0;
for(i=0;i<bytes;i+=2)
for(j=0;j<RESAMPLE_FACTOR;j++) {
resamplingBuffer[outpos++]=data[i];
resamplingBuffer[outpos++]=data[i+1];
}
WHYF("Writing %d bytes of upsampled audio",outpos);
while(played<bytes)
{
if (playMarshallBufferOffset==playBufferSize) {
/* we have a buffer full of samples, so play it */
struct msm_audio_stats stats; struct msm_audio_stats stats;
if (ioctl (playFd, AUDIO_GET_STATS, &stats) == 0) if (ioctl (playFd, AUDIO_GET_STATS, &stats) == 0)
WHYF("stats.out_bytes = %10d", stats.out_bytes); WHYF("stats.out_bytes = %10d", stats.out_bytes);
@ -489,20 +486,38 @@ int audio_msm_g1_write(unsigned char *data,int bytes)
/* even if set non-blocking the following write can block /* even if set non-blocking the following write can block
if we don't call this ioctl first */ if we don't call this ioctl first */
ioctl(playFd,AUDIO_START,0); ioctl(playFd,AUDIO_START,0);
int w=write(playFd,&resamplingBuffer[0],outpos); int w=write(playFd,&playMarshallBuffer[0],playBufferSize);
w/=RESAMPLE_FACTOR;
if (w<1) if (w<1)
{ {
WHYF("Failed to write, returned %d (errno=%s)", WHYF("Failed to write, returned %d (errno=%s)",
w,strerror(errno)); w,strerror(errno));
if (errno==EBADF) playFd=-1; if (errno==EBADF) playFd=-1;
} else { } else {
WHYF("Wrote %d bytes of audio",w); if (w<=playBufferSize) {
i+=w; /* short write, so update buffer status and inform caller */
bcopy(&playMarshallBuffer[w],&playMarshallBuffer[0],
playBufferSize-w);
playMarshallBufferOffset-=w;
WHYF("short write: %d of %d raw bytes written",
w,playBufferSize);
return w/RESAMPLE_FACTOR;
}
}
playMarshallBufferOffset=0;
} }
WHY("done writing"); /* upsample for playing back */
return bytes; for(i=0;i<RESAMPLE_FACTOR;i++) {
playMarshallBuffer[playMarshallBufferOffset++]
=data[played];
playMarshallBuffer[playMarshallBufferOffset++]
=data[played+1];
}
played+=2;
}
WHYF("done writing %d audio bytes",played);
return played;
} }
/* See if we can query end-points for this device. /* See if we can query end-points for this device.