編碼RTP over AAC的大致過程
1.添加RTP12字節的頭
2.如果AAC有ADTS頭,則跳過7字節的data
3.添加4字節的AU_HEAD
字節1:0x00
字節2:0x10
字節3:(data_len & 0x1fe0) >> 5
字節4:(data_len & 0x1f) << 3
4.添加payload(AAC的真實數據)
RTP打包圖
僞代碼
void rtp_send_aac(char *data, int len){
// skip ADTS
if(HAVE_AAC_ADTS){
len -= 7;
data += 7;
}
char buf[RTP_MAX_SIZE];
// fix buf[0~11] for rtp head
memcpy(&buf[0], &rtp_head, 12);
// fix AU_HEAD
buf[12] = 0x00;
buf[13] = 0x10;
buf[14] = (len & 0x1fe0) >> 5;
buf[15] = (len & 0x1f) << 3;
while(1){
if(default_len < len){
// fix payload
memcpy(&buf[16], data, default_len);
rtp_head.mark = 0;
rtp_send(buf, default_len + 16);
len -= default_len;
data += default_len;
}else{ // it's end
// fix last payload and set rtp mark the frame end
memcpy(&buf[16], data, len);
rtp_head.mark = 1;
rtp_send(buf, len + 16);
break;
}
}
}
RTSP->SDP
v=0
o=username 994160913 1 IN IP4
s=stream
c=IN IP4 0.0.0.0
t=0 0
a=range:npt=0-
a=sendonly
a=control:*
m=video 0 RTP/AVP 96
a=rtpmap:96 H264/90000
a=control:track0
m=audio 0 RTP/AVP 37
a=rtpmap:37 mpeg4-generic/8000/1
a=fmtp:37 streamtype=5; profile-level-id=15; mode=AAC-hbr; sizelength=13; indexlength=3; indexdeltalength=3; config=1588
a=control:track1
audio->aac->sdp->config值的計算方法:
char ConfigStr[12] = {0};
snprintf(ConfigStr, sizeof(ConfigStr), "%02x%02x",
(unsigned char)((profile + 1) << 3)|(index >> 1),
(unsigned char)((index << 7)|(channels<< 3)));
// ConfigStr爲config的值
// profile一般爲1
// channels爲音頻的通道數
// index的值需要根據採樣率確定,方法爲根據採樣率從
// 以下數組取得下表即爲index(採樣率8000即index爲11)
const unsigned int aac_samplerate[16] = {
97000, 88200, 64000, 48000,
44100, 32000, 24000, 22050,
16000, 12000, 11025, 8000,
7350, 0, 0, 0
};
注意事項
1.RTP的時間戳需要計算對,不然聲音時好時壞,時有時無。
2.AAC的SDP需要描述對,不然VLC無法正常接收。mpeg4-generic/8000/1表示採樣率爲8K,1個通道。config的值需要計算,計算方法如上描述。