diff --git a/README.md b/README.md
index 4f116f098..7761130e7 100755
--- a/README.md
+++ b/README.md
@@ -25,4 +25,16 @@ Compare:
* nginx-rtmp v1.0.4: 26786 lines
* nginx v1.5.0: 139524 lines
+Features:
+* v0.2, 2013-10-25, support h264/avc codec by rtmp complex handshake(SrsComplexHandshake).
+* v0.2, 2013-10-24, support time jitter detect and correct algorithm(SrsConsumer::jitter_correct).
+* v0.2, 2013-10-24, support decode the video/audio codec type(SrsCodec), cache the h264/avc sequence header.
+* v0.1, 2013-10-23, support basic amf0 codec, simplify the api using c-style api.
+* v0.1, 2013-10-23, support shared ptr msg(SrsSharedPtrMessage) for zero memory copy.
+* v0.1, 2013-10-22, support vp6 codec with rtmp protocol specified simple handshake.
+* v0.1, 2013-10-20, support multiple flash client play live streaming.
+* v0.1, 2013-10-20, support FMLE/FFMPEG publish live streaming.
+* v0.1, 2013-10-18, support rtmp message2chunk protocol(send message).
+* v0.1, 2013-10-17, support rtmp chunk2message protocol(recv message).
+
Winlin
diff --git a/trunk/configure b/trunk/configure
index d2512f1a0..2adacea34 100755
--- a/trunk/configure
+++ b/trunk/configure
@@ -86,7 +86,8 @@ MODULE_FILES=("srs_core" "srs_core_log" "srs_core_server"
"srs_core_error" "srs_core_conn" "srs_core_client"
"srs_core_rtmp" "srs_core_socket" "srs_core_buffer"
"srs_core_auto_free" "srs_core_protocol" "srs_core_amf0"
- "srs_core_stream" "srs_core_source" "srs_core_codec")
+ "srs_core_stream" "srs_core_source" "srs_core_codec"
+ "srs_core_complex_handshake")
MODULE_DIR="src/core" . auto/modules.sh
CORE_OBJS="${MODULE_OBJS[@]}"
diff --git a/trunk/src/core/srs_core_complex_handshake.cpp b/trunk/src/core/srs_core_complex_handshake.cpp
new file mode 100755
index 000000000..3594c8d8b
--- /dev/null
+++ b/trunk/src/core/srs_core_complex_handshake.cpp
@@ -0,0 +1,41 @@
+/*
+The MIT License (MIT)
+
+Copyright (c) 2013 winlin
+
+Permission is hereby granted, free of charge, to any person obtaining a copy of
+this software and associated documentation files (the "Software"), to deal in
+the Software without restriction, including without limitation the rights to
+use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
+the Software, and to permit persons to whom the Software is furnished to do so,
+subject to the following conditions:
+
+The above copyright notice and this permission notice shall be included in all
+copies or substantial portions of the Software.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
+FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
+COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
+IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
+CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
+*/
+
+#include
+
+#include
+
+SrsComplexHandshake::SrsComplexHandshake()
+{
+}
+
+SrsComplexHandshake::~SrsComplexHandshake()
+{
+}
+
+int SrsComplexHandshake::handshake(SrsSocket& skt, char* c1)
+{
+ int ret = ERROR_SUCCESS;
+ return ret;
+}
+
diff --git a/trunk/src/core/srs_core_complex_handshake.hpp b/trunk/src/core/srs_core_complex_handshake.hpp
new file mode 100755
index 000000000..20775cf09
--- /dev/null
+++ b/trunk/src/core/srs_core_complex_handshake.hpp
@@ -0,0 +1,58 @@
+/*
+The MIT License (MIT)
+
+Copyright (c) 2013 winlin
+
+Permission is hereby granted, free of charge, to any person obtaining a copy of
+this software and associated documentation files (the "Software"), to deal in
+the Software without restriction, including without limitation the rights to
+use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
+the Software, and to permit persons to whom the Software is furnished to do so,
+subject to the following conditions:
+
+The above copyright notice and this permission notice shall be included in all
+copies or substantial portions of the Software.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
+FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
+COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
+IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
+CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
+*/
+
+#ifndef SRS_CORE_COMPLEX_HANDSHKAE_HPP
+#define SRS_CORE_COMPLEX_HANDSHKAE_HPP
+
+/*
+#include
+*/
+
+#include
+
+class SrsSocket;
+
+/**
+* rtmp complex handshake,
+* @see also crtmp(crtmpserver) or librtmp,
+* @see also: http://blog.csdn.net/win_lin/article/details/13006803
+*/
+class SrsComplexHandshake
+{
+public:
+ SrsComplexHandshake();
+ virtual ~SrsComplexHandshake();
+public:
+ /**
+ * complex hanshake.
+ * @c1, size of c1 must be 1536.
+ * @remark, user must free the c1.
+ * @return user must:
+ * continue connect app if success,
+ * try simple handshake if error is ERROR_RTMP_TRY_SIMPLE_HS,
+ * otherwise, disconnect
+ */
+ virtual int handshake(SrsSocket& skt, char* c1);
+};
+
+#endif
\ No newline at end of file
diff --git a/trunk/src/core/srs_core_error.hpp b/trunk/src/core/srs_core_error.hpp
index 6304c1181..9807fad7d 100755
--- a/trunk/src/core/srs_core_error.hpp
+++ b/trunk/src/core/srs_core_error.hpp
@@ -62,6 +62,7 @@ CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
#define ERROR_RTMP_MESSAGE_ENCODE 308
#define ERROR_RTMP_AMF0_ENCODE 309
#define ERROR_RTMP_CHUNK_SIZE 310
+#define ERROR_RTMP_TRY_SIMPLE_HS 311
#define ERROR_SYSTEM_STREAM_INIT 400
#define ERROR_SYSTEM_PACKET_INVALID 401
diff --git a/trunk/src/core/srs_core_rtmp.cpp b/trunk/src/core/srs_core_rtmp.cpp
index 5441970a3..bd3963d87 100755
--- a/trunk/src/core/srs_core_rtmp.cpp
+++ b/trunk/src/core/srs_core_rtmp.cpp
@@ -29,6 +29,7 @@ CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
#include
#include
#include
+#include
/**
* the signature for packets to client.
@@ -139,11 +140,13 @@ SrsRtmp::SrsRtmp(st_netfd_t client_stfd)
{
protocol = new SrsProtocol(client_stfd);
stfd = client_stfd;
+ complex_handshake = new SrsComplexHandshake();
}
SrsRtmp::~SrsRtmp()
{
srs_freep(protocol);
+ srs_freep(complex_handshake);
}
void SrsRtmp::set_recv_timeout(int timeout_ms)
@@ -189,6 +192,18 @@ int SrsRtmp::handshake()
return ret;
}
srs_verbose("check c0 success, required plain text.");
+
+ // try complex handshake
+ ret = complex_handshake->handshake(skt, c0c1 + 1);
+ if (ret == ERROR_SUCCESS) {
+ srs_trace("complex handshake success.");
+ return ret;
+ }
+ if (ret != ERROR_RTMP_TRY_SIMPLE_HS) {
+ srs_error("complex handshake failed. ret=%d", ret);
+ return ret;
+ }
+ srs_info("complex handhskae failed, try simple. ret=%d", ret);
char* s0s1s2 = new char[3073];
SrsAutoFree(char, s0s1s2, true);
@@ -208,7 +223,7 @@ int SrsRtmp::handshake()
}
srs_verbose("read c2 success.");
- srs_trace("handshake success.");
+ srs_trace("simple handshake success.");
return ret;
}
diff --git a/trunk/src/core/srs_core_rtmp.hpp b/trunk/src/core/srs_core_rtmp.hpp
index 054669e84..def9727e8 100755
--- a/trunk/src/core/srs_core_rtmp.hpp
+++ b/trunk/src/core/srs_core_rtmp.hpp
@@ -39,6 +39,7 @@ class ISrsMessage;
class SrsCommonMessage;
class SrsCreateStreamPacket;
class SrsFMLEStartPacket;
+class SrsComplexHandshake;
/**
* the original request from client.
@@ -95,6 +96,7 @@ enum SrsClientType
class SrsRtmp
{
private:
+ SrsComplexHandshake* complex_handshake;
SrsProtocol* protocol;
st_netfd_t stfd;
public:
diff --git a/trunk/src/core/srs_core_source.cpp b/trunk/src/core/srs_core_source.cpp
index e39ee300b..dada68bb8 100755
--- a/trunk/src/core/srs_core_source.cpp
+++ b/trunk/src/core/srs_core_source.cpp
@@ -68,34 +68,10 @@ int SrsConsumer::enqueue(SrsSharedPtrMessage* msg)
{
int ret = ERROR_SUCCESS;
- /**
- * we use a very simple time jitter detect/correct algorithm:
- * 1. delta: ensure the delta is positive and valid,
- * we set the delta to DEFAULT_FRAME_TIME_MS,
- * if the delta of time is nagative or greater than CONST_MAX_JITTER_MS.
- * 2. last_pkt_time: specifies the original packet time,
- * is used to detect next jitter.
- * 3. last_pkt_correct_time: simply add the positive delta,
- * and enforce the time monotonically.
- */
- int32_t time = msg->header.timestamp;
- int32_t delta = time - last_pkt_time;
-
- // if jitter detected, reset the delta.
- if (delta < 0 || delta > CONST_MAX_JITTER_MS) {
- delta = DEFAULT_FRAME_TIME_MS;
-
- srs_info("jitter detected, delta=%d, last_pkt=%d, time=%d, correct_to=%d",
- delta, last_pkt_time, time, last_pkt_correct_time + delta);
- } else {
- srs_verbose("timestamp no jitter. time=%d, last_pkt=%d, correct_to=%d",
- time, last_pkt_time, last_pkt_correct_time + delta);
+ if ((ret = jitter_correct(msg)) != ERROR_SUCCESS) {
+ return ret;
}
- last_pkt_correct_time = srs_max(0, last_pkt_correct_time + delta);
- msg->header.timestamp = last_pkt_correct_time;
- last_pkt_time = time;
-
msgs.push_back(msg);
return ret;
@@ -130,6 +106,41 @@ int SrsConsumer::get_packets(int max_count, SrsSharedPtrMessage**& pmsgs, int& c
return ret;
}
+int SrsConsumer::jitter_correct(SrsSharedPtrMessage* msg)
+{
+ int ret = ERROR_SUCCESS;
+
+ /**
+ * we use a very simple time jitter detect/correct algorithm:
+ * 1. delta: ensure the delta is positive and valid,
+ * we set the delta to DEFAULT_FRAME_TIME_MS,
+ * if the delta of time is nagative or greater than CONST_MAX_JITTER_MS.
+ * 2. last_pkt_time: specifies the original packet time,
+ * is used to detect next jitter.
+ * 3. last_pkt_correct_time: simply add the positive delta,
+ * and enforce the time monotonically.
+ */
+ int32_t time = msg->header.timestamp;
+ int32_t delta = time - last_pkt_time;
+
+ // if jitter detected, reset the delta.
+ if (delta < 0 || delta > CONST_MAX_JITTER_MS) {
+ delta = DEFAULT_FRAME_TIME_MS;
+
+ srs_info("jitter detected, delta=%d, last_pkt=%d, time=%d, correct_to=%d",
+ delta, last_pkt_time, time, last_pkt_correct_time + delta);
+ } else {
+ srs_verbose("timestamp no jitter. time=%d, last_pkt=%d, correct_to=%d",
+ time, last_pkt_time, last_pkt_correct_time + delta);
+ }
+
+ last_pkt_correct_time = srs_max(0, last_pkt_correct_time + delta);
+ msg->header.timestamp = last_pkt_correct_time;
+ last_pkt_time = time;
+
+ return ret;
+}
+
SrsSource::SrsSource(std::string _stream_url)
{
stream_url = _stream_url;
diff --git a/trunk/src/core/srs_core_source.hpp b/trunk/src/core/srs_core_source.hpp
index 40419a9af..e6f2a0bc7 100755
--- a/trunk/src/core/srs_core_source.hpp
+++ b/trunk/src/core/srs_core_source.hpp
@@ -65,6 +65,11 @@ public:
* @max_count the max count to dequeue, 0 to dequeue all.
*/
virtual int get_packets(int max_count, SrsSharedPtrMessage**& pmsgs, int& count);
+private:
+ /**
+ * detect the time jitter and correct it.
+ */
+ virtual int jitter_correct(SrsSharedPtrMessage* msg);
};
/**
diff --git a/trunk/src/srs/srs.upp b/trunk/src/srs/srs.upp
index e8a64aedb..244ee4533 100755
--- a/trunk/src/srs/srs.upp
+++ b/trunk/src/srs/srs.upp
@@ -20,6 +20,8 @@ file
..\core\srs_core_codec.cpp,
..\core\srs_core_rtmp.hpp,
..\core\srs_core_rtmp.cpp,
+ ..\core\srs_core_complex_handshake.hpp,
+ ..\core\srs_core_complex_handshake.cpp,
..\core\srs_core_protocol.hpp,
..\core\srs_core_protocol.cpp,
..\core\srs_core_amf0.hpp,