forked from gegel/pairphone
-
Notifications
You must be signed in to change notification settings - Fork 0
/
tx.c
242 lines (202 loc) · 8.16 KB
/
tx.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
///////////////////////////////////////////////
//
// **************************
//
// Project/Software name: X-Phone
// Author: "Van Gegel" <[email protected]>
//
// THIS IS A FREE SOFTWARE AND FOR TEST ONLY!!!
// Please do not use it in the case of life and death
// This software is released under GNU LGPL:
//
// * LGPL 3.0 <http://www.gnu.org/licenses/lgpl.html>
//
// You’re free to copy, distribute and make commercial use
// of this software under the following conditions:
//
// * You have to cite the author (and copyright owner): Van Gegel
// * You have to provide a link to the author’s Homepage: <http://torfone.org>
//
///////////////////////////////////////////////
//This file contains transmitting procedures for PairPhone:
//We records some 8KHz audio samples (voice) from Mike, resample and collects in the buffer,
//upon sufficient melpe frame (540 samples), check frame for voice/silency,
//melpe encode or compose control silency descriptor, modulate to 3240 48KHz samples
//and play the baseband signal into Line output.
#include <math.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#ifdef _WIN32
#include <stddef.h>
#include <stdlib.h>
#include <basetsd.h>
#include <stdint.h>
#include <windows.h>
#include <time.h>
#else
#include <time.h>
#include <sys/time.h>
#include <sys/types.h>
#include <sys/stat.h>
#endif
#include "audio/audio.h" //low-level audio input/output
#include "modem/modem.h" //modem
#include "melpe/melpe.h" //audio codec
#include "vad/vad2.h" //voice active detector
#include "crp.h" //data processing
#include "tx.h" //this
vadState2 vad; //Voice Active Detector state
//recording 8 KHz speach from Mike
static short spraw[180]; //buffer for raw grabbed 8 KHz speach samples
static short spbuf[748]; //buffer for accumulate resampled voice up to melpe frame
static int spcnt=0; //number of accumulated samples
static unsigned char txbuf[12]; //buffer for encoded melpe frame or silency descryptor
//resampling
static float _up_pos=1.0; //resamplers values
static short _left_sample=0;
//playing 48 KHz baseband signal into Line
static short _jit_buf[3240]; //PCM 48KHz buffer for samples ready for playing into Line
static short* p__jit_buf=_jit_buf; //pointer to unplayed samples in the buffer
static short l__jit_buf=0; //number of unplayed samples in the buffer
//synchronizing grabbing and playing loops
static float _fdelay=24000; //average recording delay
static int tgrab=0; //difference between pointers of recording and playing streams
//*****************************************************************************
//----------------Streaming resampler--------------------------------------------
static int _resample(short* src, short* dest, int srclen, int rate)
{
//resampled srclen 8KHz samples to specified rate
//input: pointers to source and resulting short samples, number of source samples, resulting rate
//output: samples in dest resumpled from 8KHz to specified samplerate
//returns: number of resulting samples in dest
int i, diff=0;
short* sptr=src; //source
short* dptr=dest; //destination
float fstep=8000.0/rate; //ratio between specified and default rates
//process 540 samples
for(i=0;i<srclen;i++) //process samples
{
diff = *sptr-_left_sample; //computes difference beetwen current and basic samples
while(_up_pos <= 1.0) //while position not crosses a boundary
{
*dptr++ = _left_sample + ((float)diff * _up_pos); //set destination by basic, difference and position
_up_pos += fstep; //move position forward to fractional step
}
_left_sample = *sptr++; //set current sample as a basic
_up_pos = _up_pos - 1.0; //move position back to one outputted sample
}
return dptr-dest; //number of outputted samples
}
//*****************************************************************************
//--Playing over Speaker----------------------------------
static int _playjit(void)
{
//periodically try to play 8KHz samples in buffer over Speaker
int i=0;
int job=0;
if(l__jit_buf) //we have unplayed samples, try to play
{
i=_soundplay(l__jit_buf, (unsigned char*)(p__jit_buf)); //play, returns number of played samples
if(i) job+=2; //set job
if((i<0)||(i>l__jit_buf)) i=0; //must play again if underrun (PTT mode etc.)
l__jit_buf-=i; //decrease number of unplayed samples
p__jit_buf+=i; //move pointer to unplayed samples
if(l__jit_buf<=0) //all samples played
{
l__jit_buf=0; //correction
p__jit_buf=_jit_buf; //move pointer to the start of empty buffer
}
}
return job; //job flag
}
//*****************************************************************************
//transmition loop: grab 8KHz speech samples from Mike,
//resample, collect frame (540 in 67.5 mS), encode
//encrypt, modulate, play 48KHz baseband signal into Line
int tx(int job)
{
int i,j;
//loop 1: try to play unplayed samples
job+=_playjit(); //the first try to play a tail of samples in buffer
//loop 2: try to grab next 180 samples
//check for number of grabbed samples
if(spcnt<540) //we haven't enought samples for melpe encoder
{
i=soundgrab((char*)spraw, 180); //grab up to 180 samples
if((i>0)&&(i<=180)) //if some samles was grabbed
{
//Since we are using different audio devices
//on headset and line sides, the sampling rates of grabbing
// and playing devices can slightly differ then 48/8 depends HW
//so we must adjusts one of rates for synchronizing grabbing and playing processes
//The line side is more sensitive (requirements for baseband is more hard)
//That why we resamples grabbed stream (slave) for matching rate with playing stream as a master
//The adjusting process doing approximation in iterative way
//and requires several seconds for adaptation during possible loss of some speech 67.5mS frames
//computes estimated rate depends recording delay obtained in moment of last block was modulated
j=8000-(_fdelay-27000)/50; //computes samplerate using optimal delay and adjusting sensitivity
if(j>9000) j=9000; //restrict resulting samplerate
if(j<7000) j=7000;
//change rate of grabbed samples for synchronizing grabbing and playing loops
i=_resample(spraw, spbuf+spcnt, i, j); //resample and collect speech samples
spcnt+=i; //the number of samples in buffer for processing
tgrab+=i; //the total difference between grabbed speech and played baseband samples
//this is actually recording delay and must be near 270 sample in average
//for jitter protecting (due PC multi threading etc.)
job+=32; //set job
}
}
//check for we have enough grabbed samples for processing
if(spcnt>=540) //we have enough samples for melpe encoder
{
if(Mute(0)>0)
{
i=vad2(spbuf+10, &vad); //check frame is speech (by VAD)
i+=vad2(spbuf+100,&vad);
i+=vad2(spbuf+190,&vad);
i+=vad2(spbuf+280,&vad);
i+=vad2(spbuf+370,&vad);
i+=vad2(spbuf+460,&vad);
}
else i=0;
txbuf[11]=0xFF; //set defaults flag for voiced frame
if(i) //frame is voices: compress it
{
melpe_a(txbuf, spbuf); //encode the speech frame
i=State(1); //set VAD flag
}
else //unvoiced frame: sync packet will be send
{
txbuf[11]=0xFE; //or set silence flag for control blocks
i=State(-1); //clears VAD flag
}
spcnt-=540; //samples rest
if(spcnt) memcpy((char*)spbuf, (char*)(spbuf+540), 2*spcnt); //move tail to start of buffer
job+=64;
}
//Loop 3: playing
//get number of unplayed samples in buffer
i=_getdelay();
//preventing of freezing audio output after underrun or overrun
if(i>540*3*6)
{
_soundflush1();
i=_getdelay();
}
//check for delay is acceptable for playing next portion of samples
if(i<720*6)
{
if(l__jit_buf) return job; //we have some unplayed samples in local buffer, not play now.
MakePkt(txbuf); //encrypt voice or get actual control packet
l__jit_buf=Modulate(txbuf, _jit_buf); //modulate block
txbuf[11]=0; //clear tx buffer (processed)
_playjit(); //immediately play baseband into Line
//estimate rate changing for grabbed samples for synchronizing grabbing and playing
_fdelay*=0.99; //smooth coefficient
_fdelay+=tgrab; //averages recording delay
tgrab-=540; //decrease counter of grabbed samples
job+=128;
}
return job;
}