Rtmp.h 8.23 KB
Newer Older
1
/*
xiongziliang committed
2
 * Copyright (c) 2016 The ZLMediaKit project authors. All Rights Reserved.
xiongziliang committed
3
 *
4
 * This file is part of ZLMediaKit(https://github.com/xia-chu/ZLMediaKit).
xiongziliang committed
5
 *
xiongziliang committed
6 7 8
 * Use of this source code is governed by MIT license that can be found in the
 * LICENSE file in the root of the source tree. All contributing project authors
 * may be found in the AUTHORS file in the root of the source tree.
xiongziliang committed
9
 */
xiongziliang committed
10

xzl committed
11 12 13 14
#ifndef __rtmp_h
#define __rtmp_h

#include <memory>
xiongzilaing committed
15
#include <string>
16
#include <cstdlib>
xiongzilaing committed
17
#include "Util/util.h"
xzl committed
18
#include "Util/logger.h"
19
#include "Network/Buffer.h"
20
#include "Network/sockutil.h"
xiongziliang committed
21 22 23
#include "amf.h"
#include "Extension/Track.h"

xiongziliang committed
24
using namespace toolkit;
xzl committed
25

26
#if !defined(_WIN32)
xzl committed
27
#define PACKED	__attribute__((packed))
28 29 30 31 32
#else
#define PACKED
#endif //!defined(_WIN32)


xiongziliang committed
33
#define DEFAULT_CHUNK_LEN	128
xzl committed
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
#define HANDSHAKE_PLAINTEXT	0x03
#define RANDOM_LEN		(1536 - 8)

#define MSG_SET_CHUNK		1 	/*Set Chunk Size (1)*/
#define MSG_ABORT			2	/*Abort Message (2)*/
#define MSG_ACK				3 	/*Acknowledgement (3)*/
#define MSG_USER_CONTROL	4	/*User Control Messages (4)*/
#define MSG_WIN_SIZE		5	/*Window Acknowledgement Size (5)*/
#define MSG_SET_PEER_BW		6	/*Set Peer Bandwidth (6)*/
#define MSG_AUDIO			8	/*Audio Message (8)*/
#define MSG_VIDEO			9	/*Video Message (9)*/
#define MSG_DATA			18	/*Data Message (18, 15) AMF0*/
#define MSG_DATA3			15	/*Data Message (18, 15) AMF3*/
#define MSG_CMD				20	/*Command Message AMF0 */
#define MSG_CMD3			17	/*Command Message AMF3 */
#define MSG_OBJECT3			16	/*Shared Object Message (19, 16) AMF3*/
#define MSG_OBJECT			19	/*Shared Object Message (19, 16) AMF0*/
#define MSG_AGGREGATE		22	/*Aggregate Message (22)*/

#define CONTROL_STREAM_BEGIN		0
#define CONTROL_STREAM_EOF			1
#define CONTROL_STREAM_DRY			2
#define CONTROL_SETBUFFER			3
#define CONTROL_STREAM_ISRECORDED	4
#define CONTROL_PING_REQUEST		6
#define CONTROL_PING_RESPONSE		7

#define STREAM_CONTROL				0
#define STREAM_MEDIA				1

#define CHUNK_SERVER_REQUEST			2 /*服务器像客户端发出请求时的chunkID*/
#define CHUNK_CLIENT_REQUEST_BEFORE		3 /*客户端在createStream前,向服务器发出请求的chunkID*/
#define CHUNK_CLIENT_REQUEST_AFTER		4 /*客户端在createStream后,向服务器发出请求的chunkID*/
67 68
#define CHUNK_AUDIO						6 /*音频chunkID*/
#define CHUNK_VIDEO						7 /*视频chunkID*/
xzl committed
69 70 71 72

#define FLV_KEY_FRAME				1
#define FLV_INTER_FRAME				2

xiongziliang committed
73 74
#define FLV_CODEC_AAC 10
#define FLV_CODEC_H264 7
75
//金山扩展: https://github.com/ksvc/FFmpeg/wiki
xiongziliang committed
76
#define FLV_CODEC_H265 12
77 78
#define FLV_CODEC_G711A 7
#define FLV_CODEC_G711U 8
79 80
//参考学而思网校: https://github.com/notedit/rtmp/commit/6e314ac5b29611431f8fb5468596b05815743c10
#define FLV_CODEC_OPUS 13
xiongziliang committed
81

xiongziliang committed
82
namespace mediakit {
83 84 85 86 87

#if defined(_WIN32)
#pragma pack(push, 1)
#endif // defined(_WIN32)

xzl committed
88 89 90 91
class RtmpHandshake {
public:
    RtmpHandshake(uint32_t _time, uint8_t *_random = nullptr) {
        _time = htonl(_time);
xiongziliang committed
92
        memcpy(time_stamp, &_time, 4);
xzl committed
93 94 95 96 97 98
        if (!_random) {
            random_generate((char *) random, sizeof(random));
        } else {
            memcpy(random, _random, sizeof(random));
        }
    }
xiongziliang committed
99 100

    uint8_t time_stamp[4];
xiongziliang committed
101
    uint8_t zero[4] = {0};
xzl committed
102
    uint8_t random[RANDOM_LEN];
xiongziliang committed
103 104 105 106 107 108 109

    void random_generate(char *bytes, int size) {
        static char cdata[] = {0x73, 0x69, 0x6d, 0x70, 0x6c, 0x65, 0x2d, 0x72,
                               0x74, 0x6d, 0x70, 0x2d, 0x73, 0x65, 0x72, 0x76, 0x65, 0x72,
                               0x2d, 0x77, 0x69, 0x6e, 0x6c, 0x69, 0x6e, 0x2d, 0x77, 0x69,
                               0x6e, 0x74, 0x65, 0x72, 0x73, 0x65, 0x72, 0x76, 0x65, 0x72,
                               0x40, 0x31, 0x32, 0x36, 0x2e, 0x63, 0x6f, 0x6d};
xzl committed
110 111 112 113 114 115 116 117 118
        for (int i = 0; i < size; i++) {
            bytes[i] = cdata[rand() % (sizeof(cdata) - 1)];
        }
    }
}PACKED;

class RtmpHeader {
public:
    uint8_t flags;
xiongziliang committed
119 120 121 122
    uint8_t time_stamp[3];
    uint8_t body_size[3];
    uint8_t type_id;
    uint8_t stream_index[4]; /* Note, this is little-endian while others are BE */
xzl committed
123 124
}PACKED;

125 126 127
#if defined(_WIN32)
#pragma pack(pop)
#endif // defined(_WIN32)
xzl committed
128

129
class RtmpPacket : public Buffer{
xzl committed
130
public:
xia-chu committed
131
    friend class RtmpProtocol;
xia-chu committed
132 133
    using Ptr = std::shared_ptr<RtmpPacket>;
    bool is_abs_stamp;
xiongziliang committed
134
    uint8_t type_id;
xia-chu committed
135 136
    uint32_t time_stamp;
    uint32_t ts_field;
xiongziliang committed
137 138
    uint32_t stream_index;
    uint32_t chunk_id;
xia-chu committed
139
    size_t body_size;
140
    BufferLikeString buffer;
xiongziliang committed
141

142
public:
xia-chu committed
143 144
    static Ptr create();

145
    char *data() const override{
xiongziliang committed
146
        return (char*)buffer.data();
147
    }
148
    size_t size() const override {
xiongziliang committed
149 150 151
        return buffer.size();
    }

xia-chu committed
152 153 154 155 156 157
    void clear(){
        is_abs_stamp = false;
        time_stamp = 0;
        ts_field = 0;
        body_size = 0;
        buffer.clear();
xiongziliang committed
158
    }
xiongziliang committed
159

xzl committed
160
    bool isVideoKeyFrame() const {
xiongziliang committed
161
        return type_id == MSG_VIDEO && (uint8_t) buffer[0] >> 4 == FLV_KEY_FRAME && (uint8_t) buffer[1] == 1;
xzl committed
162
    }
xiongziliang committed
163

xzl committed
164
    bool isCfgFrame() const {
xiongziliang committed
165 166
        switch (type_id){
            case MSG_VIDEO : return buffer[1] == 0;
xiongziliang committed
167 168
            case MSG_AUDIO : {
                switch (getMediaType()){
xiongziliang committed
169
                    case FLV_CODEC_AAC : return buffer[1] == 0;
xiongziliang committed
170 171 172 173 174
                    default : return false;
                }
            }
            default : return false;
        }
xzl committed
175
    }
xiongziliang committed
176

xzl committed
177
    int getMediaType() const {
xiongziliang committed
178 179 180
        switch (type_id) {
            case MSG_VIDEO : return (uint8_t) buffer[0] & 0x0F;
            case MSG_AUDIO : return (uint8_t) buffer[0] >> 4;
xiongziliang committed
181
            default : return 0;
xzl committed
182 183
        }
    }
xiongziliang committed
184

xzl committed
185
    int getAudioSampleRate() const {
xiongziliang committed
186
        if (type_id != MSG_AUDIO) {
xzl committed
187 188
            return 0;
        }
xiongziliang committed
189
        int flvSampleRate = ((uint8_t) buffer[0] & 0x0C) >> 2;
xzl committed
190 191 192
        const static int sampleRate[] = { 5512, 11025, 22050, 44100 };
        return sampleRate[flvSampleRate];
    }
xiongziliang committed
193

xzl committed
194
    int getAudioSampleBit() const {
xiongziliang committed
195
        if (type_id != MSG_AUDIO) {
xzl committed
196 197
            return 0;
        }
xiongziliang committed
198
        int flvSampleBit = ((uint8_t) buffer[0] & 0x02) >> 1;
xzl committed
199 200 201
        const static int sampleBit[] = { 8, 16 };
        return sampleBit[flvSampleBit];
    }
xiongziliang committed
202

xzl committed
203
    int getAudioChannel() const {
xiongziliang committed
204
        if (type_id != MSG_AUDIO) {
xzl committed
205 206
            return 0;
        }
xiongziliang committed
207
        int flvStereoOrMono = (uint8_t) buffer[0] & 0x01;
xzl committed
208 209 210
        const static int channel[] = { 1, 2 };
        return channel[flvStereoOrMono];
    }
xia-chu committed
211 212 213 214 215 216

private:
    friend class ResourcePool_l<RtmpPacket>;
    RtmpPacket(){
        clear();
    }
xia-chu committed
217

xia-chu committed
218 219 220 221 222 223 224 225 226 227
    RtmpPacket &operator=(const RtmpPacket &that) {
        is_abs_stamp = that.is_abs_stamp;
        stream_index = that.stream_index;
        body_size = that.body_size;
        type_id = that.type_id;
        ts_field = that.ts_field;
        time_stamp = that.time_stamp;
        return *this;
    }

xia-chu committed
228 229 230
private:
    //对象个数统计
    ObjectStatistic<RtmpPacket> _statistic;
xzl committed
231 232
};

xiongziliang committed
233
/**
234
 * rtmp metadata基类,用于描述rtmp格式信息
xiongziliang committed
235
 */
236
class Metadata : public CodecInfo{
xiongziliang committed
237
public:
238
    typedef std::shared_ptr<Metadata> Ptr;
xiongziliang committed
239

240 241 242 243
    Metadata():_metadata(AMF_OBJECT){}
    virtual ~Metadata(){}
    const AMFValue &getMetadata() const{
        return _metadata;
xiongziliang committed
244
    }
245 246

    static void addTrack(AMFValue &metadata, const Track::Ptr &track);
xiongziliang committed
247
protected:
248
    AMFValue _metadata;
xiongziliang committed
249 250 251
};

/**
252
* metadata中除音视频外的其他描述部分
xiongziliang committed
253
*/
254
class TitleMeta : public Metadata{
xiongziliang committed
255
public:
256
    typedef std::shared_ptr<TitleMeta> Ptr;
xiongziliang committed
257

258
    TitleMeta(float dur_sec = 0,
259
              size_t fileSize = 0,
xiongziliang committed
260
              const map<string,string> &header = map<string,string>()){
261 262
        _metadata.set("duration", dur_sec);
        _metadata.set("fileSize", 0);
xiongziliang committed
263
        _metadata.set("server",SERVER_NAME);
xiongziliang committed
264
        for (auto &pr : header){
265
            _metadata.set(pr.first, pr.second);
xiongziliang committed
266 267 268 269 270 271 272 273
        }
    }

    CodecId getCodecId() const override{
        return CodecInvalid;
    }
};

274
class VideoMeta : public Metadata{
xiongziliang committed
275
public:
276
    typedef std::shared_ptr<VideoMeta> Ptr;
xiongziliang committed
277

xiongziliang committed
278
    VideoMeta(const VideoTrack::Ptr &video);
279
    virtual ~VideoMeta(){}
xiongziliang committed
280 281 282 283 284 285 286 287

    CodecId getCodecId() const override{
        return _codecId;
    }
private:
    CodecId _codecId;
};

288
class AudioMeta : public Metadata{
xiongziliang committed
289
public:
290
    typedef std::shared_ptr<AudioMeta> Ptr;
xiongziliang committed
291

xiongziliang committed
292
    AudioMeta(const AudioTrack::Ptr &audio);
xiongziliang committed
293

294
    virtual ~AudioMeta(){}
xiongziliang committed
295 296 297 298 299 300 301 302

    CodecId getCodecId() const override{
        return _codecId;
    }
private:
    CodecId _codecId;
};

xiongziliang committed
303 304
//根据音频track获取flags
uint8_t getAudioRtmpFlags(const Track::Ptr &track);
xiongziliang committed
305

xiongziliang committed
306
}//namespace mediakit
xiongziliang committed
307
#endif//__rtmp_h