Rtmp.h 8.12 KB
Newer Older
1
/*
xiongziliang committed
2
 * Copyright (c) 2016 The ZLMediaKit project authors. All Rights Reserved.
xiongziliang committed
3 4 5
 *
 * This file is part of ZLMediaKit(https://github.com/xiongziliang/ZLMediaKit).
 *
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 131
public:
    typedef std::shared_ptr<RtmpPacket> Ptr;
xiongziliang committed
132 133 134 135 136 137 138
    uint8_t type_id;
    uint32_t body_size = 0;
    uint32_t time_stamp = 0;
    bool is_abs_stamp = false;
    uint32_t ts_field = 0;
    uint32_t stream_index;
    uint32_t chunk_id;
139
    BufferLikeString buffer;
xiongziliang committed
140

141 142
public:
    char *data() const override{
xiongziliang committed
143
        return (char*)buffer.data();
144 145
    }
    uint32_t size() const override {
xiongziliang committed
146 147 148
        return buffer.size();
    }

xiongziliang committed
149 150
public:
    RtmpPacket() = default;
151 152 153
    RtmpPacket(const RtmpPacket &that) = delete;
    RtmpPacket &operator=(const RtmpPacket &that) = delete;
    RtmpPacket &operator=(RtmpPacket &&that) = delete;
xiongziliang committed
154 155

    RtmpPacket(RtmpPacket &&that){
xiongziliang committed
156 157 158 159 160 161 162 163
        type_id = that.type_id;
        body_size = that.body_size;
        time_stamp = that.time_stamp;
        is_abs_stamp = that.is_abs_stamp;
        ts_field = that.ts_field;
        stream_index = that.stream_index;
        chunk_id = that.chunk_id;
        buffer = std::move(that.buffer);
xiongziliang committed
164
    }
xiongziliang committed
165

xzl committed
166
    bool isVideoKeyFrame() const {
xiongziliang committed
167
        return type_id == MSG_VIDEO && (uint8_t) buffer[0] >> 4 == FLV_KEY_FRAME && (uint8_t) buffer[1] == 1;
xzl committed
168
    }
xiongziliang committed
169

xzl committed
170
    bool isCfgFrame() const {
xiongziliang committed
171 172
        switch (type_id){
            case MSG_VIDEO : return buffer[1] == 0;
xiongziliang committed
173 174
            case MSG_AUDIO : {
                switch (getMediaType()){
xiongziliang committed
175
                    case FLV_CODEC_AAC : return buffer[1] == 0;
xiongziliang committed
176 177 178 179 180
                    default : return false;
                }
            }
            default : return false;
        }
xzl committed
181
    }
xiongziliang committed
182

xzl committed
183
    int getMediaType() const {
xiongziliang committed
184 185 186
        switch (type_id) {
            case MSG_VIDEO : return (uint8_t) buffer[0] & 0x0F;
            case MSG_AUDIO : return (uint8_t) buffer[0] >> 4;
xiongziliang committed
187
            default : return 0;
xzl committed
188 189
        }
    }
xiongziliang committed
190

xzl committed
191
    int getAudioSampleRate() const {
xiongziliang committed
192
        if (type_id != MSG_AUDIO) {
xzl committed
193 194
            return 0;
        }
xiongziliang committed
195
        int flvSampleRate = ((uint8_t) buffer[0] & 0x0C) >> 2;
xzl committed
196 197 198
        const static int sampleRate[] = { 5512, 11025, 22050, 44100 };
        return sampleRate[flvSampleRate];
    }
xiongziliang committed
199

xzl committed
200
    int getAudioSampleBit() const {
xiongziliang committed
201
        if (type_id != MSG_AUDIO) {
xzl committed
202 203
            return 0;
        }
xiongziliang committed
204
        int flvSampleBit = ((uint8_t) buffer[0] & 0x02) >> 1;
xzl committed
205 206 207
        const static int sampleBit[] = { 8, 16 };
        return sampleBit[flvSampleBit];
    }
xiongziliang committed
208

xzl committed
209
    int getAudioChannel() const {
xiongziliang committed
210
        if (type_id != MSG_AUDIO) {
xzl committed
211 212
            return 0;
        }
xiongziliang committed
213
        int flvStereoOrMono = (uint8_t) buffer[0] & 0x01;
xzl committed
214 215 216 217 218
        const static int channel[] = { 1, 2 };
        return channel[flvStereoOrMono];
    }
};

xiongziliang committed
219
/**
220
 * rtmp metadata基类,用于描述rtmp格式信息
xiongziliang committed
221
 */
222
class Metadata : public CodecInfo{
xiongziliang committed
223
public:
224
    typedef std::shared_ptr<Metadata> Ptr;
xiongziliang committed
225

226 227 228 229
    Metadata():_metadata(AMF_OBJECT){}
    virtual ~Metadata(){}
    const AMFValue &getMetadata() const{
        return _metadata;
xiongziliang committed
230
    }
231 232

    static void addTrack(AMFValue &metadata, const Track::Ptr &track);
xiongziliang committed
233
protected:
234
    AMFValue _metadata;
xiongziliang committed
235 236 237
};

/**
238
* metadata中除音视频外的其他描述部分
xiongziliang committed
239
*/
240
class TitleMeta : public Metadata{
xiongziliang committed
241
public:
242
    typedef std::shared_ptr<TitleMeta> Ptr;
xiongziliang committed
243

244
    TitleMeta(float dur_sec = 0,
xiongziliang committed
245 246
              uint64_t fileSize = 0,
              const map<string,string> &header = map<string,string>()){
247 248
        _metadata.set("duration", dur_sec);
        _metadata.set("fileSize", 0);
xiongziliang committed
249
        _metadata.set("server",SERVER_NAME);
xiongziliang committed
250
        for (auto &pr : header){
251
            _metadata.set(pr.first, pr.second);
xiongziliang committed
252 253 254 255 256 257 258 259
        }
    }

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

260
class VideoMeta : public Metadata{
xiongziliang committed
261
public:
262
    typedef std::shared_ptr<VideoMeta> Ptr;
xiongziliang committed
263

xiongziliang committed
264
    VideoMeta(const VideoTrack::Ptr &video);
265
    virtual ~VideoMeta(){}
xiongziliang committed
266 267 268 269 270 271 272 273

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

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

xiongziliang committed
278
    AudioMeta(const AudioTrack::Ptr &audio);
xiongziliang committed
279

280
    virtual ~AudioMeta(){}
xiongziliang committed
281 282 283 284 285 286 287 288

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

xiongziliang committed
289 290
//根据音频track获取flags
uint8_t getAudioRtmpFlags(const Track::Ptr &track);
xiongziliang committed
291

xiongziliang committed
292
}//namespace mediakit
xiongziliang committed
293
#endif//__rtmp_h