Skip to content
GitLab
Projects
Groups
Snippets
Help
Loading...
Help
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in
Toggle navigation
E
ESP32-audioI2S
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Analytics
Analytics
Repository
Value Stream
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Commits
Open sidebar
xpstem
ESP32-audioI2S
Commits
6793db0b
Unverified
Commit
6793db0b
authored
Jul 28, 2022
by
Wolle
Committed by
GitHub
Jul 28, 2022
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
more hls (ts) streams works now
parent
6aece4dc
Changes
2
Hide whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
149 additions
and
100 deletions
+149
-100
src/Audio.cpp
src/Audio.cpp
+146
-98
src/Audio.h
src/Audio.h
+3
-2
No files found.
src/Audio.cpp
View file @
6793db0b
...
@@ -3,8 +3,8 @@
...
@@ -3,8 +3,8 @@
*
*
* Created on: Oct 26.2018
* Created on: Oct 26.2018
*
*
* Version 2.0.5
* Version 2.0.5
a
* Updated on: Jul 2
5
.2022
* Updated on: Jul 2
8
.2022
* Author: Wolle (schreibfaul1)
* Author: Wolle (schreibfaul1)
*
*
*/
*/
...
@@ -356,6 +356,7 @@ void Audio::setDefaults() {
...
@@ -356,6 +356,7 @@ void Audio::setDefaults() {
m_streamTitleHash
=
0
;
m_streamTitleHash
=
0
;
m_file_size
=
0
;
m_file_size
=
0
;
m_m3u8_timeStamp
=
0
;
m_m3u8_timeStamp
=
0
;
m_ID3Size
=
0
;
}
}
//---------------------------------------------------------------------------------------------------------------------
//---------------------------------------------------------------------------------------------------------------------
...
@@ -1411,6 +1412,7 @@ int Audio::read_ID3_Header(uint8_t *data, size_t len) {
...
@@ -1411,6 +1412,7 @@ int Audio::read_ID3_Header(uint8_t *data, size_t len) {
m_controlCounter
=
10
;
m_controlCounter
=
10
;
}
}
headerSize
=
id3Size
;
headerSize
=
id3Size
;
m_ID3Size
=
id3Size
;
headerSize
-=
10
;
headerSize
-=
10
;
return
10
;
return
10
;
...
@@ -2221,11 +2223,11 @@ void Audio::loop() {
...
@@ -2221,11 +2223,11 @@ void Audio::loop() {
if
(
m_playlistFormat
==
FORMAT_ASX
)
host
=
parsePlaylist_ASX
();
if
(
m_playlistFormat
==
FORMAT_ASX
)
host
=
parsePlaylist_ASX
();
if
(
host
){
if
(
host
){
if
(
m_playlistFormat
==
FORMAT_M3U8
)
m_f_m3u8data
=
true
;
if
(
m_playlistFormat
==
FORMAT_M3U8
)
m_f_m3u8data
=
true
;
if
(
m_f_m3u8data
)
AUDIO_INFO
(
"cth %s"
,
host
);
if
(
!
_client
){
stopSong
();
log_e
(
"no client found!"
);
return
;}
connecttohost
(
host
);
connecttohost
(
host
);
}
}
else
{
else
{
if
(
m_f_ts
)
m_m3u8_timeStamp
=
millis
()
+
m_m3u8_targetDuration
*
85
0
;
// sower because unpack ts frames
if
(
m_f_ts
)
m_m3u8_timeStamp
=
millis
()
+
m_m3u8_targetDuration
*
60
0
;
// sower because unpack ts frames
else
m_m3u8_timeStamp
=
millis
()
+
m_m3u8_targetDuration
*
1200
;
else
m_m3u8_timeStamp
=
millis
()
+
m_m3u8_targetDuration
*
1200
;
setDatamode
(
AUDIO_DATA
);
//fake datamode, we have no new audiosequence yet, so let audio run
setDatamode
(
AUDIO_DATA
);
//fake datamode, we have no new audiosequence yet, so let audio run
}
}
...
@@ -2292,7 +2294,7 @@ bool Audio::readPlayListData() {
...
@@ -2292,7 +2294,7 @@ bool Audio::readPlayListData() {
if
(
startsWith
(
pl
,
"<!DOCTYPE"
))
{
AUDIO_INFO
(
"url is a webpage!"
);
goto
exit
;}
if
(
startsWith
(
pl
,
"<!DOCTYPE"
))
{
AUDIO_INFO
(
"url is a webpage!"
);
goto
exit
;}
if
(
strlen
(
pl
)
>
0
)
m_playlistContent
.
push_back
(
strdup
((
const
char
*
)
pl
));
if
(
strlen
(
pl
)
>
0
)
m_playlistContent
.
push_back
(
strdup
((
const
char
*
)
pl
));
if
(
m_playlistContent
.
size
()
==
100
){
if
(
m_playlistContent
.
size
()
==
100
){
ESP_LOGD
(
""
,
"the maximum number of lines in the playlist has been reached"
);
log_d
(
"the maximum number of lines in the playlist has been reached"
);
break
;
break
;
}
}
if
(
ctl
==
m_contentlength
){
break
;}
if
(
ctl
==
m_contentlength
){
break
;}
...
@@ -2301,7 +2303,7 @@ bool Audio::readPlayListData() {
...
@@ -2301,7 +2303,7 @@ bool Audio::readPlayListData() {
}
// outer while6nUfOrsqhhT-331
}
// outer while6nUfOrsqhhT-331
lines
=
m_playlistContent
.
size
();
lines
=
m_playlistContent
.
size
();
for
(
int
i
=
0
;
i
<
lines
;
i
++
)
{
// print all string in first vector of 'arr'
for
(
int
i
=
0
;
i
<
lines
;
i
++
)
{
// print all string in first vector of 'arr'
ESP_LOGD
(
"pl="
,
"
%i
\"
%s
\"
"
,
i
,
m_playlistContent
[
i
]);
log_d
(
"pl=
%i
\"
%s
\"
"
,
i
,
m_playlistContent
[
i
]);
}
}
m_datamode
=
AUDIO_PLAYLISTDATA
;
m_datamode
=
AUDIO_PLAYLISTDATA
;
return
true
;
return
true
;
...
@@ -2507,7 +2509,7 @@ const char* Audio::parsePlaylist_M3U8(){
...
@@ -2507,7 +2509,7 @@ const char* Audio::parsePlaylist_M3U8(){
targetDuration
=
atoi
(
m_playlistContent
[
i
]
+
22
);
targetDuration
=
atoi
(
m_playlistContent
[
i
]
+
22
);
}
}
if
(
targetDuration
)
m_m3u8_targetDuration
=
targetDuration
;
if
(
targetDuration
)
m_m3u8_targetDuration
=
targetDuration
;
// log_e
("m_m3u8_targetDuration %d", m_m3u8_targetDuration);
log_d
(
"m_m3u8_targetDuration %d"
,
m_m3u8_targetDuration
);
if
(
startsWith
(
m_playlistContent
[
i
],
"#EXTINF"
))
{
if
(
startsWith
(
m_playlistContent
[
i
],
"#EXTINF"
))
{
if
(
STfromEXTINF
(
m_playlistContent
[
i
]))
showstreamtitle
(
chbuf
);
if
(
STfromEXTINF
(
m_playlistContent
[
i
]))
showstreamtitle
(
chbuf
);
...
@@ -2528,12 +2530,14 @@ const char* Audio::parsePlaylist_M3U8(){
...
@@ -2528,12 +2530,14 @@ const char* Audio::parsePlaylist_M3U8(){
if
(
!
m_m3u8_lastEntry
){
// first init
if
(
!
m_m3u8_lastEntry
){
// first init
m_playlistURL
.
insert
(
m_playlistURL
.
begin
(),
strdup
((
const
char
*
)(
tmp
)));
m_playlistURL
.
insert
(
m_playlistURL
.
begin
(),
strdup
((
const
char
*
)(
tmp
)));
m_m3u8_lastEntry
=
strdup
(
tmp
);
m_m3u8_lastEntry
=
strdup
(
tmp
);
log_d
(
"insert %s"
,
tmp
);
continue
;
continue
;
}
}
if
(
strcmp
(
tmp
,
m_m3u8_lastEntry
)
>
0
){
// next sequence?,
if
(
strcmp
(
tmp
,
m_m3u8_lastEntry
)
>
0
){
// next sequence?,
m_playlistURL
.
insert
(
m_playlistURL
.
begin
(),
strdup
((
const
char
*
)(
tmp
)));
m_playlistURL
.
insert
(
m_playlistURL
.
begin
(),
strdup
((
const
char
*
)(
tmp
)));
if
(
m_m3u8_lastEntry
){
free
(
m_m3u8_lastEntry
);
m_m3u8_lastEntry
=
strdup
(
tmp
);}
if
(
m_m3u8_lastEntry
){
free
(
m_m3u8_lastEntry
);
m_m3u8_lastEntry
=
strdup
(
tmp
);}
assert
(
m_m3u8_lastEntry
!=
tmp
);
// no free space in task?
assert
(
m_m3u8_lastEntry
!=
tmp
);
// no free space in task?
log_d
(
"insert %s"
,
tmp
);
}
}
else
{
else
{
AUDIO_INFO
(
"file already known %s"
,
m_playlistContent
[
i
]);
AUDIO_INFO
(
"file already known %s"
,
m_playlistContent
[
i
]);
...
@@ -3058,7 +3062,6 @@ void Audio::processWebStreamTS() {
...
@@ -3058,7 +3062,6 @@ void Audio::processWebStreamTS() {
// first call, set some values to default - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
// first call, set some values to default - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
if
(
m_f_firstCall
)
{
// runs only ont time per connection, prepare for start
if
(
m_f_firstCall
)
{
// runs only ont time per connection, prepare for start
m_f_firstCall
=
false
;
f_stream
=
false
;
f_stream
=
false
;
byteCounter
=
0
;
byteCounter
=
0
;
bytesDecoded
=
0
;
bytesDecoded
=
0
;
...
@@ -3066,6 +3069,26 @@ void Audio::processWebStreamTS() {
...
@@ -3066,6 +3069,26 @@ void Audio::processWebStreamTS() {
tmr_1s
=
millis
();
tmr_1s
=
millis
();
m_t0
=
millis
();
m_t0
=
millis
();
ts_packetPtr
=
0
;
ts_packetPtr
=
0
;
ts_parsePacket
(
0
,
0
,
0
);
// reset ts routine
if
(
m_contentlength
%
188
!=
0
){
// For files without an ID3 header, the content length is always divided by 188 without a remainder
static
size_t
ID3frameSize
=
0
;
uint8_t
ID3headerLength
=
10
;
if
(
!
ID3frameSize
){
uint8_t
ID3headerLength
=
10
;
if
(
_client
->
available
()
<
ID3headerLength
)
return
;
_client
->
read
(
ts_packet
,
ID3headerLength
);
read_ID3_Header
(
ts_packet
,
ID3headerLength
);
ID3frameSize
=
m_ID3Size
;
}
if
(
ID3frameSize
>
188
)
{
log_e
(
"ID3 Header too long"
);
stopSong
();
return
;}
if
(
_client
->
available
()
<
ID3frameSize
-
ID3headerLength
)
return
;
_client
->
read
(
ts_packet
,
ID3frameSize
-
ID3headerLength
);
read_ID3_Header
(
ts_packet
,
ID3frameSize
-
ID3headerLength
);
byteCounter
+=
ID3frameSize
;
ID3frameSize
=
0
;
}
m_f_firstCall
=
false
;
}
}
if
(
m_datamode
!=
AUDIO_DATA
)
return
;
// guard
if
(
m_datamode
!=
AUDIO_DATA
)
return
;
// guard
...
@@ -3194,14 +3217,14 @@ bool Audio::parseHttpResponseHeader() { // this is the response to a GET / reque
...
@@ -3194,14 +3217,14 @@ bool Audio::parseHttpResponseHeader() { // this is the response to a GET / reque
pos
++
;
pos
++
;
if
(
pos
==
510
){
if
(
pos
==
510
){
rhl
[
pos
]
=
'\0'
;
rhl
[
pos
]
=
'\0'
;
ESP_LOGD
(
""
,
"responseHeaderline overflow"
);
log_d
(
"responseHeaderline overflow"
);
break
;
break
;
}
}
}
// inner while
}
// inner while
if
(
!
pos
)
{
vTaskDelay
(
3
);
continue
;}
if
(
!
pos
)
{
vTaskDelay
(
3
);
continue
;}
if
(
m_f_Log
)
{
AUDIO_INFO
(
"httpResponseHeader: %s"
,
rhl
);}
if
(
m_f_Log
)
{
log_d
(
"httpResponseHeader: %s"
,
rhl
);}
int16_t
posColon
=
indexOf
(
rhl
,
":"
,
0
);
// lowercase all letters up to the colon
int16_t
posColon
=
indexOf
(
rhl
,
":"
,
0
);
// lowercase all letters up to the colon
if
(
posColon
>=
0
)
{
if
(
posColon
>=
0
)
{
...
@@ -3219,7 +3242,6 @@ bool Audio::parseHttpResponseHeader() { // this is the response to a GET / reque
...
@@ -3219,7 +3242,6 @@ bool Audio::parseHttpResponseHeader() { // this is the response to a GET / reque
statusCode
[
3
]
=
'\0'
;
statusCode
[
3
]
=
'\0'
;
int
sc
=
atoi
(
statusCode
);
int
sc
=
atoi
(
statusCode
);
if
(
sc
>
310
||
sc
==
301
){
// e.g. HTTP/1.1 301 Moved Permanently
if
(
sc
>
310
||
sc
==
301
){
// e.g. HTTP/1.1 301 Moved Permanently
AUDIO_INFO
(
rhl
);
if
(
audio_showstreamtitle
)
audio_showstreamtitle
(
rhl
);
if
(
audio_showstreamtitle
)
audio_showstreamtitle
(
rhl
);
// goto exit;
// goto exit;
}
}
...
@@ -3359,12 +3381,14 @@ bool Audio::parseHttpResponseHeader() { // this is the response to a GET / reque
...
@@ -3359,12 +3381,14 @@ bool Audio::parseHttpResponseHeader() { // this is the response to a GET / reque
if
(
m_codec
!=
CODEC_NONE
){
if
(
m_codec
!=
CODEC_NONE
){
m_datamode
=
AUDIO_DATA
;
// Expecting data now
m_datamode
=
AUDIO_DATA
;
// Expecting data now
if
(
!
initializeDecoder
())
return
false
;
if
(
!
initializeDecoder
())
return
false
;
if
(
m_f_Log
)
{
AUDIO_INFO
(
"Switch to DATA, metaint is %d"
,
m_metaint
);}
if
(
m_f_Log
)
{
log_i
(
"Switch to DATA, metaint is %d"
,
m_metaint
);}
if
(
m_playlistFormat
!=
FORMAT_M3U8
&&
audio_lasthost
)
audio_lasthost
(
m_lastHost
);
if
(
m_playlistFormat
!=
FORMAT_M3U8
&&
audio_lasthost
)
audio_lasthost
(
m_lastHost
);
m_controlCounter
=
0
;
m_f_firstCall
=
true
;
}
}
else
if
(
m_playlistFormat
!=
FORMAT_NONE
){
else
if
(
m_playlistFormat
!=
FORMAT_NONE
){
m_datamode
=
AUDIO_PLAYLISTINIT
;
// playlist expected
m_datamode
=
AUDIO_PLAYLISTINIT
;
// playlist expected
if
(
m_f_Log
)
{
AUDIO_INFO
(
"now parse playlist"
);}
if
(
m_f_Log
)
{
log_i
(
"now parse playlist"
);}
}
}
else
{
else
{
goto
exit
;
goto
exit
;
...
@@ -3525,27 +3549,27 @@ bool Audio::parseContentType(char* ct) {
...
@@ -3525,27 +3549,27 @@ bool Audio::parseContentType(char* ct) {
switch
(
ct_val
){
switch
(
ct_val
){
case
CT_MP3
:
case
CT_MP3
:
m_codec
=
CODEC_MP3
;
m_codec
=
CODEC_MP3
;
if
(
m_f_Log
)
{
AUDIO_INFO
(
"ContentType %s, format is mp3"
,
ct
);
}
//ok is likely mp3
if
(
m_f_Log
)
{
log_i
(
"ContentType %s, format is mp3"
,
ct
);
}
//ok is likely mp3
break
;
break
;
case
CT_AAC
:
case
CT_AAC
:
m_codec
=
CODEC_AAC
;
m_codec
=
CODEC_AAC
;
if
(
m_f_Log
)
{
AUDIO_INFO
(
"ContentType %s, format is aac"
,
ct
);
}
if
(
m_f_Log
)
{
log_i
(
"ContentType %s, format is aac"
,
ct
);
}
break
;
break
;
case
CT_M4A
:
case
CT_M4A
:
m_codec
=
CODEC_M4A
;
m_codec
=
CODEC_M4A
;
if
(
m_f_Log
)
{
AUDIO_INFO
(
"ContentType %s, format is aac"
,
ct
);
}
if
(
m_f_Log
)
{
log_i
(
"ContentType %s, format is aac"
,
ct
);
}
break
;
break
;
case
CT_FLAC
:
case
CT_FLAC
:
m_codec
=
CODEC_FLAC
;
m_codec
=
CODEC_FLAC
;
if
(
m_f_Log
)
{
AUDIO_INFO
(
"ContentType %s, format is flac"
,
ct
);
}
if
(
m_f_Log
)
{
log_i
(
"ContentType %s, format is flac"
,
ct
);
}
break
;
break
;
case
CT_WAV
:
case
CT_WAV
:
m_codec
=
CODEC_WAV
;
m_codec
=
CODEC_WAV
;
if
(
m_f_Log
)
{
AUDIO_INFO
(
"ContentType %s, format is wav"
,
ct
);
}
if
(
m_f_Log
)
{
log_i
(
"ContentType %s, format is wav"
,
ct
);
}
break
;
break
;
case
CT_OGG
:
case
CT_OGG
:
m_codec
=
CODEC_OGG
;
m_codec
=
CODEC_OGG
;
if
(
m_f_Log
)
{
AUDIO_INFO
(
"ContentType %s found"
,
ct
);
}
if
(
m_f_Log
)
{
log_i
(
"ContentType %s found"
,
ct
);
}
break
;
break
;
case
CT_PLS
:
case
CT_PLS
:
...
@@ -4495,47 +4519,77 @@ bool Audio::ts_parsePacket(uint8_t* packet, uint8_t* packetStart, uint8_t* packe
...
@@ -4495,47 +4519,77 @@ bool Audio::ts_parsePacket(uint8_t* packet, uint8_t* packetStart, uint8_t* packe
const
u_int8_t
PID_ARRAY_LEN
=
4
;
const
u_int8_t
PID_ARRAY_LEN
=
4
;
typedef
struct
{
typedef
struct
{
int
number
=
0
;
int
number
=
0
;
int
pids
[
PID_ARRAY_LEN
];
int
pids
[
PID_ARRAY_LEN
];
}
pid_array
;
}
pid_array
;
static
pid_array
pidsOfPMT
,
pidsOfAAC
;
static
pid_array
pidsOfPMT
;
static
uint16_t
pesDataLength
=
0
;
static
int
PES_DataLength
=
0
;
static
int
pidOfAAC
=
0
;
if
(
packet
==
NULL
){
log_i
(
"reset"
);
for
(
int
i
=
0
;
i
<
PID_ARRAY_LEN
;
i
++
)
pidsOfPMT
.
pids
[
i
]
=
0
;
PES_DataLength
=
0
;
pidOfAAC
=
0
;
return
true
;
}
// --------------------------------------------------------------------------------------------------------
// 0. Byte SyncByte | 0 | 1 | 0 | 0 | 0 | 1 | 1 | 1 | always bit pattern of 0x47
//---------------------------------------------------------------------------------------------------------
// 1. Byte |PUSI|TP| |PID|PID|PID|PID|PID|
//---------------------------------------------------------------------------------------------------------
// 2. Byte |PID|PID|PID|PID|PID|PID|PID|PID|
//---------------------------------------------------------------------------------------------------------
// 3. Byte |TSC|TSC|AFC|ADC|CC |CC |CC |CC |
//---------------------------------------------------------------------------------------------------------
// 4.-187. Byte |Payload data if AFC==01 or 11 |
//---------------------------------------------------------------------------------------------------------
// PUSI Payload unit start indicator, set when this packet contains the first byte of a new payload unit.
// The first byte of the payload will indicate where this new payload unit starts.
// TP Transport priority, set when the current packet has a higher priority than other packets with the same PID.
// PID Packet Identifier, describing the payload data.
// TSC Transport scrambling control, '00' = Not scrambled.
// AFC Adaptation field control, 01 – no adaptation field, payload only, 10 – adaptation field only, no payload,
// 11 – adaptation field followed by payload, 00 – RESERVED for future use
// CC Continuity counter, Sequence number of payload packets (0x00 to 0x0F) within each stream (except PID 8191)
if
(
packet
[
0
]
!=
0x47
)
{
if
(
packet
[
0
]
!=
0x47
)
{
log_e
(
"ts sync byte not found"
);
log_e
(
"ts SyncByte not found, first bytes are %X %X %X %X"
,
packet
[
0
],
packet
[
1
],
packet
[
2
],
packet
[
3
]);
stopSong
();
return
false
;
return
false
;
}
}
int
PID
=
(
packet
[
1
]
&
0x1F
)
<<
8
|
(
packet
[
2
]
&
0xFF
);
if
(
debug
)
printf
(
"PID: 0x%04X(%d)
\n
"
,
PID
,
PID
);
int
PUSI
=
(
packet
[
1
]
&
0x40
)
>>
6
;
if
(
debug
)
printf
(
"Payload Unit Start Indicator: %d
\n
"
,
PUSI
);
int
AFC
=
(
packet
[
3
]
&
0x30
)
>>
4
;
if
(
debug
)
printf
(
"Adaption Field Control: %d
\n
"
,
AFC
);
int
pid
=
(
packet
[
1
]
&
0x1F
)
<<
8
|
(
packet
[
2
]
&
0xFF
);
int
AFL
=
-
1
;
if
(
debug
)
printf
(
"PID: 0x%04X(%d)
\n
"
,
pid
,
pid
);
if
((
AFC
&
0b10
)
==
0b10
)
{
// AFC '11' Adaptation Field followed
int
pusi
=
(
packet
[
1
]
&
0x40
)
>>
6
;
AFL
=
packet
[
4
]
&
0xFF
;
// Adaptation Field Length
if
(
debug
)
printf
(
"Payload Unit Start Indicator: %d
\n
"
,
pusi
);
if
(
debug
)
printf
(
"Adaptation Field Length: %d
\n
"
,
AFL
);
int
afc
=
(
packet
[
3
]
&
0x30
)
>>
4
;
if
(
debug
)
printf
(
"Adaption Field Control: %d
\n
"
,
afc
);
int
rafl
=
-
1
;
// remaining Adaptation Field Length
if
((
afc
&
0b10
)
==
0b10
)
{
rafl
=
packet
[
4
]
&
0xFF
;
if
(
debug
)
printf
(
"Adaptation Field Length: %d
\n
"
,
rafl
);
}
}
int
pls
=
pusi
?
5
:
4
;
// payloadStart, payloadUnitStartInc
icator
int
PLS
=
PUSI
?
5
:
4
;
// PayLoadStart, Payload Unit Start Ind
icator
if
(
pid
==
0
)
{
if
(
PID
==
0
)
{
// PAT
// Program Association Table (PAT) - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
// log_i("PAT");
if
(
debug
)
printf
(
"PAT
\n
"
);
if
(
debug
)
printf
(
"PAT
\n
"
);
pidsOfPMT
.
number
=
0
;
pidsOfPMT
.
number
=
0
;
pidsOfAAC
.
number
=
0
;
pidOfAAC
=
0
;
pesDataLength
=
0
;
int
startOfProgramNums
=
8
;
int
startOfProgramNums
=
8
;
int
lengthOfPATValue
=
4
;
int
lengthOfPATValue
=
4
;
int
sectionLength
=
((
packet
[
pls
+
1
]
&
0x0F
)
<<
8
)
|
(
packet
[
pls
+
2
]
&
0xFF
);
int
sectionLength
=
((
packet
[
PLS
+
1
]
&
0x0F
)
<<
8
)
|
(
packet
[
PLS
+
2
]
&
0xFF
);
if
(
debug
)
printf
(
"Section Length: %d
\n
"
,
sectionLength
);
if
(
debug
)
printf
(
"Section Length: %d
\n
"
,
sectionLength
);
int
program_number
,
program_map_PID
;
int
program_number
,
program_map_PID
;
int
indexOfPids
=
0
;
int
indexOfPids
=
0
;
for
(
int
i
=
startOfProgramNums
;
i
<=
sectionLength
;
i
+=
lengthOfPATValue
)
{
for
(
int
i
=
startOfProgramNums
;
i
<=
sectionLength
;
i
+=
lengthOfPATValue
)
{
program_number
=
((
packet
[
pls
+
i
]
&
0xFF
)
<<
8
)
|
(
packet
[
pls
+
i
+
1
]
&
0xFF
);
program_number
=
((
packet
[
PLS
+
i
]
&
0xFF
)
<<
8
)
|
(
packet
[
PLS
+
i
+
1
]
&
0xFF
);
program_map_PID
=
((
packet
[
pls
+
i
+
2
]
&
0x1F
)
<<
8
)
|
(
packet
[
pls
+
i
+
3
]
&
0xFF
);
program_map_PID
=
((
packet
[
PLS
+
i
+
2
]
&
0x1F
)
<<
8
)
|
(
packet
[
PLS
+
i
+
3
]
&
0xFF
);
if
(
debug
)
printf
(
"Program Num: 0x%04X(%d) PMT PID: 0x%04X(%d)
\n
"
,
program_number
,
program_number
,
if
(
debug
)
printf
(
"Program Num: 0x%04X(%d) PMT PID: 0x%04X(%d)
\n
"
,
program_number
,
program_number
,
program_map_PID
,
program_map_PID
);
program_map_PID
,
program_map_PID
);
pidsOfPMT
.
pids
[
indexOfPids
++
]
=
program_map_PID
;
pidsOfPMT
.
pids
[
indexOfPids
++
]
=
program_map_PID
;
...
@@ -4546,87 +4600,81 @@ bool Audio::ts_parsePacket(uint8_t* packet, uint8_t* packetStart, uint8_t* packe
...
@@ -4546,87 +4600,81 @@ bool Audio::ts_parsePacket(uint8_t* packet, uint8_t* packetStart, uint8_t* packe
return
true
;
return
true
;
}
}
else
if
(
pidsOfAAC
.
number
)
{
else
if
(
PID
==
pidOfAAC
)
{
for
(
int
i
=
0
;
i
<
pidsOfAAC
.
number
;
i
++
)
{
if
(
debug
)
printf
(
"AAC
\n
"
);
if
(
pid
==
pidsOfAAC
.
pids
[
i
])
{
uint8_t
posOfPacketStart
=
4
;
if
(
debug
)
printf
(
"AAC
\n
"
);
if
(
AFL
>=
0
)
{
posOfPacketStart
=
5
+
AFL
;
uint8_t
posOfPacketStart
=
4
;
if
(
debug
)
printf
(
"posOfPacketStart: %d
\n
"
,
posOfPacketStart
);}
if
(
rafl
>=
0
)
{
posOfPacketStart
=
5
+
rafl
;
if
(
debug
)
printf
(
"posOfPacketStart: %d
\n
"
,
posOfPacketStart
);}
// Packetized Elementary Stream (PES) - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
if
(
PES_DataLength
>
0
)
{
size_t
dataSize
;
*
packetStart
=
posOfPacketStart
;
if
(
pesDataLength
>
0
)
{
*
packetLength
=
TS_PACKET_SIZE
-
posOfPacketStart
;
dataSize
=
TS_PACKET_SIZE
-
posOfPacketStart
;
PES_DataLength
-=
TS_PACKET_SIZE
-
posOfPacketStart
;
*
packetStart
=
posOfPacketStart
;
return
true
;
*
packetLength
=
dataSize
;
}
pesDataLength
-=
dataSize
;
else
{
return
true
;
int
firstByte
=
packet
[
posOfPacketStart
]
&
0xFF
;
}
int
secondByte
=
packet
[
posOfPacketStart
+
1
]
&
0xFF
;
else
{
int
thirdByte
=
packet
[
posOfPacketStart
+
2
]
&
0xFF
;
int
firstByte
=
packet
[
posOfPacketStart
]
&
0xFF
;
if
(
debug
)
printf
(
"First 3 bytes: %02X %02X %02X
\n
"
,
firstByte
,
secondByte
,
thirdByte
);
int
secondByte
=
packet
[
posOfPacketStart
+
1
]
&
0xFF
;
if
(
firstByte
==
0x00
&&
secondByte
==
0x00
&&
thirdByte
==
0x01
)
{
// Packet start code prefix
int
thirdByte
=
packet
[
posOfPacketStart
+
2
]
&
0xFF
;
// PES
if
(
debug
)
printf
(
"First 3 bytes: %02X %02X %02X
\n
"
,
firstByte
,
secondByte
,
thirdByte
);
uint8_t
StreamID
=
packet
[
posOfPacketStart
+
3
]
&
0xFF
;
if
(
firstByte
==
0x00
&&
secondByte
==
0x00
&&
thirdByte
==
0x01
)
{
if
(
StreamID
>=
0xC0
&&
StreamID
<=
0xDF
)
{;}
// okay ist audio stream
// PES
if
(
StreamID
>=
0xE0
&&
StreamID
<=
0xEF
)
{
log_e
(
"video stream!"
);
return
false
;}
uint8_t
streamID
=
packet
[
posOfPacketStart
+
3
]
&
0xFF
;
const
uint8_t
posOfPacketLengthLatterHalf
=
5
;
if
(
streamID
>=
0xC0
&&
streamID
<=
0xDF
)
{;}
// okay ist audio stream
int
PES_PacketLength
=
if
(
streamID
>=
0xE0
&&
streamID
<=
0xEF
)
{
log_e
(
"video stream!"
);
return
false
;}
((
packet
[
posOfPacketStart
+
4
]
&
0xFF
)
<<
8
)
+
(
packet
[
posOfPacketStart
+
5
]
&
0xFF
);
const
uint8_t
posOfPacketLengthLatterHalf
=
5
;
if
(
debug
)
printf
(
"PES Packet length: %d
\n
"
,
PES_PacketLength
);
int
PES_PacketLength
=
PES_DataLength
=
PES_PacketLength
;
((
packet
[
posOfPacketStart
+
4
]
&
0xFF
)
<<
8
)
+
(
packet
[
posOfPacketStart
+
5
]
&
0xFF
);
int
posOfHeaderLength
=
8
;
if
(
debug
)
printf
(
"PES Packet length: %d
\n
"
,
PES_PacketLength
);
int
PESRemainingHeaderLength
=
packet
[
posOfPacketStart
+
posOfHeaderLength
]
&
0xFF
;
pesDataLength
=
PES_PacketLength
;
if
(
debug
)
printf
(
"PES Header length: %d
\n
"
,
PESRemainingHeaderLength
);
int
posOfHeaderLength
=
8
;
int
startOfData
=
posOfHeaderLength
+
PESRemainingHeaderLength
+
1
;
int
PESRemainingHeaderLength
=
packet
[
posOfPacketStart
+
posOfHeaderLength
]
&
0xFF
;
if
(
debug
)
printf
(
"First AAC data byte: %02X
\n
"
,
packet
[
posOfPacketStart
+
startOfData
]);
if
(
debug
)
printf
(
"PES Header length: %d
\n
"
,
PESRemainingHeaderLength
);
*
packetStart
=
posOfPacketStart
+
startOfData
;
int
startOfData
=
posOfHeaderLength
+
PESRemainingHeaderLength
+
1
;
*
packetLength
=
TS_PACKET_SIZE
-
posOfPacketStart
-
startOfData
;
if
(
debug
)
printf
(
"First AAC data byte: %02X
\n
"
,
packet
[
posOfPacketStart
+
startOfData
]);
PES_DataLength
-=
(
TS_PACKET_SIZE
-
posOfPacketStart
)
-
(
posOfPacketLengthLatterHalf
+
1
);
*
packetStart
=
posOfPacketStart
+
startOfData
;
*
packetLength
=
TS_PACKET_SIZE
-
posOfPacketStart
-
startOfData
;
pesDataLength
-=
(
TS_PACKET_SIZE
-
posOfPacketStart
)
-
(
posOfPacketLengthLatterHalf
+
1
);
return
true
;
}
}
*
packetStart
=
0
;
*
packetLength
=
0
;
return
true
;
return
true
;
}
}
}
}
*
packetStart
=
0
;
*
packetLength
=
0
;
// log_e("PES not found");
return
false
;
}
}
else
if
(
pidsOfPMT
.
number
)
{
else
if
(
pidsOfPMT
.
number
)
{
// Program Map Table (PMT) - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
// log_i("PMT");
for
(
int
i
=
0
;
i
<
pidsOfPMT
.
number
;
i
++
)
{
for
(
int
i
=
0
;
i
<
pidsOfPMT
.
number
;
i
++
)
{
if
(
pid
==
pidsOfPMT
.
pids
[
i
])
{
if
(
PID
==
pidsOfPMT
.
pids
[
i
])
{
if
(
debug
)
printf
(
"PMT
\n
"
);
if
(
debug
)
printf
(
"PMT
\n
"
);
int
staticLengthOfPMT
=
12
;
int
staticLengthOfPMT
=
12
;
int
sectionLength
=
((
packet
[
pls
+
1
]
&
0x0F
)
<<
8
)
|
(
packet
[
pls
+
2
]
&
0xFF
);
int
sectionLength
=
((
packet
[
PLS
+
1
]
&
0x0F
)
<<
8
)
|
(
packet
[
PLS
+
2
]
&
0xFF
);
if
(
debug
)
printf
(
"Section Length: %d
\n
"
,
sectionLength
);
if
(
debug
)
printf
(
"Section Length: %d
\n
"
,
sectionLength
);
int
programInfoLength
=
((
packet
[
pls
+
10
]
&
0x0F
)
<<
8
)
|
(
packet
[
pls
+
11
]
&
0xFF
);
int
programInfoLength
=
((
packet
[
PLS
+
10
]
&
0x0F
)
<<
8
)
|
(
packet
[
PLS
+
11
]
&
0xFF
);
if
(
debug
)
printf
(
"Program Info Length: %d
\n
"
,
programInfoLength
);
if
(
debug
)
printf
(
"Program Info Length: %d
\n
"
,
programInfoLength
);
int
indexOfPids
=
pidsOfAAC
.
number
;
int
cursor
=
staticLengthOfPMT
+
programInfoLength
;
int
cursor
=
staticLengthOfPMT
+
programInfoLength
;
while
(
cursor
<
sectionLength
-
1
)
{
while
(
cursor
<
sectionLength
-
1
)
{
int
streamType
=
packet
[
pls
+
cursor
]
&
0xFF
;
int
streamType
=
packet
[
PLS
+
cursor
]
&
0xFF
;
int
elementaryPID
=
((
packet
[
pls
+
cursor
+
1
]
&
0x1F
)
<<
8
)
|
(
packet
[
pls
+
cursor
+
2
]
&
0xFF
);
int
elementaryPID
=
((
packet
[
PLS
+
cursor
+
1
]
&
0x1F
)
<<
8
)
|
(
packet
[
PLS
+
cursor
+
2
]
&
0xFF
);
if
(
debug
)
printf
(
"Stream Type: 0x%02X Elementary PID: 0x%04X
\n
"
,
streamType
,
elementaryPID
);
if
(
debug
)
printf
(
"Stream Type: 0x%02X Elementary PID: 0x%04X
\n
"
,
streamType
,
elementaryPID
);
if
(
streamType
==
0x0F
||
streamType
==
0x11
)
{
if
(
streamType
==
0x0F
||
streamType
==
0x11
)
{
if
(
debug
)
printf
(
"AAC PID discover
\n
"
);
if
(
debug
)
printf
(
"AAC PID discover
\n
"
);
pid
sOfAAC
.
pids
[
indexOfPids
++
]
=
elementaryPID
;
pid
OfAAC
=
elementaryPID
;
}
}
int
esInfoLength
=
((
packet
[
pls
+
cursor
+
3
]
&
0x0F
)
<<
8
)
|
(
packet
[
pls
+
cursor
+
4
]
&
0xFF
);
int
esInfoLength
=
((
packet
[
PLS
+
cursor
+
3
]
&
0x0F
)
<<
8
)
|
(
packet
[
PLS
+
cursor
+
4
]
&
0xFF
);
if
(
debug
)
printf
(
"ES Info Length: 0x%04X
\n
"
,
esInfoLength
);
if
(
debug
)
printf
(
"ES Info Length: 0x%04X
\n
"
,
esInfoLength
);
cursor
+=
5
+
esInfoLength
;
cursor
+=
5
+
esInfoLength
;
}
}
pidsOfAAC
.
number
=
indexOfPids
;
}
}
}
}
*
packetStart
=
0
;
*
packetStart
=
0
;
*
packetLength
=
0
;
*
packetLength
=
0
;
return
true
;
return
true
;
}
}
return
true
;
// log_e("invalid ts packet!");
return
false
;
}
}
//----------------------------------------------------------------------------------------------------------------------
//----------------------------------------------------------------------------------------------------------------------
//----------------------------------------------------------------------------------------------------------------------
src/Audio.h
View file @
6793db0b
/*
/*
* Audio.h
* Audio.h
*
*
* Created on: Oct 2
6
,2018
* Created on: Oct 2
8
,2018
* Updated on: Jul 25,2022
* Updated on: Jul 25,2022
* Author: Wolle (schreibfaul1)
* Author: Wolle (schreibfaul1)
*/
*/
...
@@ -410,7 +410,7 @@ private:
...
@@ -410,7 +410,7 @@ private:
M4A_ILST
=
7
,
M4A_MP4A
=
8
,
M4A_AMRDY
=
99
,
M4A_OKAY
=
100
};
M4A_ILST
=
7
,
M4A_MP4A
=
8
,
M4A_AMRDY
=
99
,
M4A_OKAY
=
100
};
enum
:
int
{
OGG_BEGIN
=
0
,
OGG_MAGIC
=
1
,
OGG_HEADER
=
2
,
OGG_FIRST
=
3
,
OGG_AMRDY
=
99
,
OGG_OKAY
=
100
};
enum
:
int
{
OGG_BEGIN
=
0
,
OGG_MAGIC
=
1
,
OGG_HEADER
=
2
,
OGG_FIRST
=
3
,
OGG_AMRDY
=
99
,
OGG_OKAY
=
100
};
enum
:
int
{
CODEC_NONE
=
0
,
CODEC_WAV
=
1
,
CODEC_MP3
=
2
,
CODEC_AAC
=
3
,
CODEC_M4A
=
4
,
CODEC_FLAC
=
5
,
enum
:
int
{
CODEC_NONE
=
0
,
CODEC_WAV
=
1
,
CODEC_MP3
=
2
,
CODEC_AAC
=
3
,
CODEC_M4A
=
4
,
CODEC_FLAC
=
5
,
CODEC_OGG
=
6
,
CODEC_OGG_FLAC
=
7
,
CODEC_OGG_OPUS
=
8
};
CODEC_OGG
=
6
,
CODEC_OGG_FLAC
=
7
,
CODEC_OGG_OPUS
=
8
,
CODEC_AACP
=
9
};
enum
:
int
{
ST_NONE
=
0
,
ST_WEBFILE
=
1
,
ST_WEBSTREAM
=
2
};
enum
:
int
{
ST_NONE
=
0
,
ST_WEBFILE
=
1
,
ST_WEBSTREAM
=
2
};
typedef
enum
{
LEFTCHANNEL
=
0
,
RIGHTCHANNEL
=
1
}
SampleIndex
;
typedef
enum
{
LEFTCHANNEL
=
0
,
RIGHTCHANNEL
=
1
}
SampleIndex
;
typedef
enum
{
LOWSHELF
=
0
,
PEAKEQ
=
1
,
HIFGSHELF
=
2
}
FilterType
;
typedef
enum
{
LOWSHELF
=
0
,
PEAKEQ
=
1
,
HIFGSHELF
=
2
}
FilterType
;
...
@@ -473,6 +473,7 @@ private:
...
@@ -473,6 +473,7 @@ private:
uint8_t
m_expectedPlsFmt
=
FORMAT_NONE
;
// set in connecttohost (e.g. streaming01.m3u) -> FORMAT_M3U)
uint8_t
m_expectedPlsFmt
=
FORMAT_NONE
;
// set in connecttohost (e.g. streaming01.m3u) -> FORMAT_M3U)
uint8_t
m_filterType
[
2
];
// lowpass, highpass
uint8_t
m_filterType
[
2
];
// lowpass, highpass
uint8_t
m_streamType
=
ST_NONE
;
uint8_t
m_streamType
=
ST_NONE
;
uint8_t
m_ID3Size
=
0
;
// lengt of ID3frame - ID3header
int16_t
m_outBuff
[
2048
*
2
];
// Interleaved L/R
int16_t
m_outBuff
[
2048
*
2
];
// Interleaved L/R
int16_t
m_validSamples
=
0
;
int16_t
m_validSamples
=
0
;
int16_t
m_curSample
=
0
;
int16_t
m_curSample
=
0
;
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment