Rtmp.h 7.95 KB
Newer Older
xiongziliang committed
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 27

#define PORT	1935
#define DEFAULT_CHUNK_LEN	128
28 29

#if !defined(_WIN32)
xzl committed
30
#define PACKED	__attribute__((packed))
31 32 33 34 35
#else
#define PACKED
#endif //!defined(_WIN32)


xzl committed
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 67 68
#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*/
69 70
#define CHUNK_AUDIO						6 /*音频chunkID*/
#define CHUNK_VIDEO						7 /*视频chunkID*/
xzl committed
71 72 73 74

#define FLV_KEY_FRAME				1
#define FLV_INTER_FRAME				2

xiongziliang committed
75 76 77
#define FLV_CODEC_AAC 10
#define FLV_CODEC_H264 7
#define FLV_CODEC_H265 12
78 79 80
#define FLV_CODEC_G711A 7
#define FLV_CODEC_G711U 8

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 92 93 94 95 96 97 98 99
class RtmpHandshake {
public:
    RtmpHandshake(uint32_t _time, uint8_t *_random = nullptr) {
        _time = htonl(_time);
        memcpy(timeStamp, &_time, 4);
        if (!_random) {
            random_generate((char *) random, sizeof(random));
        } else {
            memcpy(random, _random, sizeof(random));
        }
    }
    uint8_t timeStamp[4];
xiongziliang committed
100
    uint8_t zero[4] = {0};
xzl committed
101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122
    uint8_t random[RANDOM_LEN];
    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 };
        for (int i = 0; i < size; i++) {
            bytes[i] = cdata[rand() % (sizeof(cdata) - 1)];
        }
    }
}PACKED;

class RtmpHeader {
public:
    uint8_t flags;
    uint8_t timeStamp[3];
    uint8_t bodySize[3];
    uint8_t typeId;
    uint8_t streamId[4]; /* Note, this is little-endian while others are BE */
}PACKED;

123 124 125
#if defined(_WIN32)
#pragma pack(pop)
#endif // defined(_WIN32)
xzl committed
126

127
class RtmpPacket : public Buffer{
xzl committed
128 129 130 131 132
public:
    typedef std::shared_ptr<RtmpPacket> Ptr;
    uint8_t typeId;
    uint32_t bodySize = 0;
    uint32_t timeStamp = 0;
xzl committed
133
    bool hasAbsStamp = false;
xzl committed
134
    bool hasExtStamp = false;
xzl committed
135
    uint32_t deltaStamp = 0;
xzl committed
136 137 138
    uint32_t streamId;
    uint32_t chunkId;
    std::string strBuf;
139 140 141 142 143 144 145
public:
    char *data() const override{
        return (char*)strBuf.data();
    }
    uint32_t size() const override {
        return strBuf.size();
    };
xiongziliang committed
146 147
public:
    RtmpPacket() = default;
148 149 150
    RtmpPacket(const RtmpPacket &that) = delete;
    RtmpPacket &operator=(const RtmpPacket &that) = delete;
    RtmpPacket &operator=(RtmpPacket &&that) = delete;
xiongziliang committed
151 152 153 154 155 156 157 158 159 160 161 162

    RtmpPacket(RtmpPacket &&that){
        typeId = that.typeId;
        bodySize = that.bodySize;
        timeStamp = that.timeStamp;
        hasAbsStamp = that.hasAbsStamp;
        hasExtStamp = that.hasExtStamp;
        deltaStamp = that.deltaStamp;
        streamId = that.streamId;
        chunkId = that.chunkId;
        strBuf = std::move(that.strBuf);
    }
xzl committed
163
    bool isVideoKeyFrame() const {
xiongziliang committed
164
        return typeId == MSG_VIDEO && (uint8_t) strBuf[0] >> 4 == FLV_KEY_FRAME && (uint8_t) strBuf[1] == 1;
xzl committed
165 166
    }
    bool isCfgFrame() const {
xiongziliang committed
167 168 169 170 171 172 173 174 175 176
        switch (typeId){
            case MSG_VIDEO : return strBuf[1] == 0;
            case MSG_AUDIO : {
                switch (getMediaType()){
                    case FLV_CODEC_AAC : return strBuf[1] == 0;
                    default : return false;
                }
            }
            default : return false;
        }
xzl committed
177 178 179
    }
    int getMediaType() const {
        switch (typeId) {
xiongziliang committed
180 181 182
            case MSG_VIDEO : return (uint8_t) strBuf[0] & 0x0F;
            case MSG_AUDIO : return (uint8_t) strBuf[0] >> 4;
            default : return 0;
xzl committed
183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210
        }
    }
    int getAudioSampleRate() const {
        if (typeId != MSG_AUDIO) {
            return 0;
        }
        int flvSampleRate = ((uint8_t) strBuf[0] & 0x0C) >> 2;
        const static int sampleRate[] = { 5512, 11025, 22050, 44100 };
        return sampleRate[flvSampleRate];
    }
    int getAudioSampleBit() const {
        if (typeId != MSG_AUDIO) {
            return 0;
        }
        int flvSampleBit = ((uint8_t) strBuf[0] & 0x02) >> 1;
        const static int sampleBit[] = { 8, 16 };
        return sampleBit[flvSampleBit];
    }
    int getAudioChannel() const {
        if (typeId != MSG_AUDIO) {
            return 0;
        }
        int flvStereoOrMono = (uint8_t) strBuf[0] & 0x01;
        const static int channel[] = { 1, 2 };
        return channel[flvStereoOrMono];
    }
};

xiongziliang committed
211
/**
212
 * rtmp metadata基类,用于描述rtmp格式信息
xiongziliang committed
213
 */
214
class Metadata : public CodecInfo{
xiongziliang committed
215
public:
216
    typedef std::shared_ptr<Metadata> Ptr;
xiongziliang committed
217

218 219 220 221
    Metadata():_metadata(AMF_OBJECT){}
    virtual ~Metadata(){}
    const AMFValue &getMetadata() const{
        return _metadata;
xiongziliang committed
222
    }
223 224

    static void addTrack(AMFValue &metadata, const Track::Ptr &track);
xiongziliang committed
225
protected:
226
    AMFValue _metadata;
xiongziliang committed
227 228 229
};

/**
230
* metadata中除音视频外的其他描述部分
xiongziliang committed
231
*/
232
class TitleMeta : public Metadata{
xiongziliang committed
233
public:
234
    typedef std::shared_ptr<TitleMeta> Ptr;
xiongziliang committed
235

236
    TitleMeta(float dur_sec = 0,
xiongziliang committed
237 238
              uint64_t fileSize = 0,
              const map<string,string> &header = map<string,string>()){
239 240
        _metadata.set("duration", dur_sec);
        _metadata.set("fileSize", 0);
xiongziliang committed
241
        _metadata.set("server",SERVER_NAME);
xiongziliang committed
242
        for (auto &pr : header){
243
            _metadata.set(pr.first, pr.second);
xiongziliang committed
244 245 246 247 248 249 250 251
        }
    }

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

252
class VideoMeta : public Metadata{
xiongziliang committed
253
public:
254
    typedef std::shared_ptr<VideoMeta> Ptr;
xiongziliang committed
255

256 257
    VideoMeta(const VideoTrack::Ptr &video,int datarate = 5000);
    virtual ~VideoMeta(){}
xiongziliang committed
258 259 260 261 262 263 264 265

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

266
class AudioMeta : public Metadata{
xiongziliang committed
267
public:
268
    typedef std::shared_ptr<AudioMeta> Ptr;
xiongziliang committed
269

270
    AudioMeta(const AudioTrack::Ptr &audio,int datarate = 160);
xiongziliang committed
271

272
    virtual ~AudioMeta(){}
xiongziliang committed
273 274 275 276 277 278 279 280

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

xiongziliang committed
281 282
//根据音频track获取flags
uint8_t getAudioRtmpFlags(const Track::Ptr &track);
xiongziliang committed
283

xiongziliang committed
284
}//namespace mediakit
xiongziliang committed
285
#endif//__rtmp_h