ZLMediaKit/src/Rtmp/Rtmp.h

274 lines
6.8 KiB
C++
Raw Normal View History

2020-08-01 10:22:12 +08:00
/*
2020-04-04 20:30:09 +08:00
* Copyright (c) 2016 The ZLMediaKit project authors. All Rights Reserved.
2017-09-27 16:20:30 +08:00
*
* This file is part of ZLMediaKit(https://github.com/xia-chu/ZLMediaKit).
2017-09-27 16:20:30 +08:00
*
2020-04-04 20:30:09 +08:00
* 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.
2017-09-27 16:20:30 +08:00
*/
2020-04-04 20:30:09 +08:00
2017-04-01 16:35:56 +08:00
#ifndef __rtmp_h
#define __rtmp_h
#include <memory>
2017-04-25 11:35:41 +08:00
#include <string>
2019-01-16 14:26:06 +08:00
#include <cstdlib>
2019-06-28 17:25:53 +08:00
#include "amf.h"
#include "Network/Buffer.h"
2019-06-28 17:25:53 +08:00
#include "Extension/Track.h"
2017-08-09 18:39:30 +08:00
#if !defined(_WIN32)
2017-04-01 16:35:56 +08:00
#define PACKED __attribute__((packed))
2017-08-09 18:39:30 +08:00
#else
#define PACKED
#endif //!defined(_WIN32)
2020-08-30 10:48:34 +08:00
#define DEFAULT_CHUNK_LEN 128
2017-04-01 16:35:56 +08:00
#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_NETWORK 2 /*网络相关的消息(参见 Protocol Control Messages)*/
#define CHUNK_SYSTEM 3 /*向服务器发送控制消息(反之亦可)*/
2017-04-01 16:35:56 +08:00
#define CHUNK_CLIENT_REQUEST_BEFORE 3 /*客户端在createStream前,向服务器发出请求的chunkID*/
#define CHUNK_CLIENT_REQUEST_AFTER 4 /*客户端在createStream后,向服务器发出请求的chunkID*/
#define CHUNK_AUDIO 6 /*音频chunkID*/
#define CHUNK_VIDEO 7 /*视频chunkID*/
2017-04-01 16:35:56 +08:00
#define FLV_KEY_FRAME 1
#define FLV_INTER_FRAME 2
2020-04-04 22:54:49 +08:00
#define FLV_CODEC_AAC 10
#define FLV_CODEC_H264 7
2020-08-01 10:22:12 +08:00
//金山扩展: https://github.com/ksvc/FFmpeg/wiki
2020-04-04 22:54:49 +08:00
#define FLV_CODEC_H265 12
#define FLV_CODEC_G711A 7
#define FLV_CODEC_G711U 8
2020-08-01 10:22:12 +08:00
//参考学而思网校: https://github.com/notedit/rtmp/commit/6e314ac5b29611431f8fb5468596b05815743c10
#define FLV_CODEC_OPUS 13
2020-04-04 22:54:49 +08:00
2019-03-27 18:41:52 +08:00
namespace mediakit {
2017-08-09 18:39:30 +08:00
#if defined(_WIN32)
#pragma pack(push, 1)
#endif // defined(_WIN32)
2017-04-01 16:35:56 +08:00
class RtmpHandshake {
public:
RtmpHandshake(uint32_t _time, uint8_t *_random = nullptr);
2020-08-30 10:48:34 +08:00
uint8_t time_stamp[4];
2017-05-13 17:25:31 +08:00
uint8_t zero[4] = {0};
2017-04-01 16:35:56 +08:00
uint8_t random[RANDOM_LEN];
2020-08-30 10:48:34 +08:00
void random_generate(char *bytes, int size);
void create_complex_c0c1();
2017-04-01 16:35:56 +08:00
}PACKED;
class RtmpHeader {
public:
2021-06-22 10:39:16 +08:00
#if __BYTE_ORDER == __BIG_ENDIAN
uint8_t fmt : 2;
uint8_t chunk_id : 6;
#else
uint8_t chunk_id : 6;
//0、1、2、3分别对应 12、8、4、1长度
uint8_t fmt : 2;
#endif
2020-08-30 10:48:34 +08:00
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 */
2017-04-01 16:35:56 +08:00
}PACKED;
2021-07-15 11:16:11 +08:00
class FLVHeader {
public:
2023-06-17 10:29:27 +08:00
static constexpr uint8_t kFlvVersion = 1;
static constexpr uint8_t kFlvHeaderLength = 9;
2021-07-15 11:16:11 +08:00
//FLV
char flv[3];
//File version (for example, 0x01 for FLV version 1)
uint8_t version;
#if __BYTE_ORDER == __BIG_ENDIAN
//保留,置0
uint8_t : 5;
//是否有音频
uint8_t have_audio: 1;
//保留,置0
uint8_t : 1;
//是否有视频
uint8_t have_video: 1;
#else
//是否有视频
uint8_t have_video: 1;
//保留,置0
uint8_t : 1;
//是否有音频
uint8_t have_audio: 1;
//保留,置0
uint8_t : 5;
#endif
//The length of this header in bytes,固定为9
uint32_t length;
2023-06-17 10:29:27 +08:00
//固定为0
uint32_t previous_tag_size0;
2021-07-15 11:16:11 +08:00
} PACKED;
class RtmpTagHeader {
public:
uint8_t type = 0;
uint8_t data_size[3] = {0};
uint8_t timestamp[3] = {0};
uint8_t timestamp_ex = 0;
uint8_t streamid[3] = {0}; /* Always 0. */
} PACKED;
2017-08-09 18:39:30 +08:00
#if defined(_WIN32)
#pragma pack(pop)
#endif // defined(_WIN32)
2017-04-01 16:35:56 +08:00
class RtmpPacket : public toolkit::Buffer{
2017-04-01 16:35:56 +08:00
public:
2021-02-07 23:01:22 +08:00
friend class RtmpProtocol;
2021-02-04 17:58:51 +08:00
using Ptr = std::shared_ptr<RtmpPacket>;
bool is_abs_stamp;
2020-08-30 10:48:34 +08:00
uint8_t type_id;
2021-02-04 17:58:51 +08:00
uint32_t time_stamp;
uint32_t ts_field;
2020-08-30 10:48:34 +08:00
uint32_t stream_index;
uint32_t chunk_id;
2021-02-04 17:58:51 +08:00
size_t body_size;
toolkit::BufferLikeString buffer;
2020-08-30 10:48:34 +08:00
public:
2021-02-04 17:58:51 +08:00
static Ptr create();
char *data() const override{
2020-08-30 10:48:34 +08:00
return (char*)buffer.data();
}
size_t size() const override {
2020-08-30 10:48:34 +08:00
return buffer.size();
}
void clear();
2020-08-30 10:48:34 +08:00
bool isVideoKeyFrame() const;
bool isCfgFrame() const;
2020-08-30 10:48:34 +08:00
int getMediaType() const;
2020-08-30 10:48:34 +08:00
int getAudioSampleRate() const;
int getAudioSampleBit() const;
int getAudioChannel() const;
2021-02-04 17:58:51 +08:00
private:
friend class toolkit::ResourcePool_l<RtmpPacket>;
2021-02-04 17:58:51 +08:00
RtmpPacket(){
clear();
}
2021-02-05 16:49:11 +08:00
RtmpPacket &operator=(const RtmpPacket &that);
2021-02-07 23:01:22 +08:00
2021-02-05 16:49:11 +08:00
private:
//对象个数统计
toolkit::ObjectStatistic<RtmpPacket> _statistic;
2017-04-01 16:35:56 +08:00
};
2019-06-28 17:25:53 +08:00
/**
* rtmp metadata基类rtmp格式信息
2019-06-28 17:25:53 +08:00
*/
class Metadata : public CodecInfo{
2019-06-28 17:25:53 +08:00
public:
2022-12-02 14:43:06 +08:00
using Ptr = std::shared_ptr<Metadata>;
2019-06-28 17:25:53 +08:00
Metadata(): _metadata(AMF_OBJECT) {}
virtual ~Metadata() = default;
const AMFValue &getMetadata() const{
return _metadata;
2019-06-28 17:25:53 +08:00
}
static void addTrack(AMFValue &metadata, const Track::Ptr &track);
2019-06-28 17:25:53 +08:00
protected:
AMFValue _metadata;
2019-06-28 17:25:53 +08:00
};
/**
* metadata中除音视频外的其他描述部分
2019-06-28 17:25:53 +08:00
*/
class TitleMeta : public Metadata{
2019-06-28 17:25:53 +08:00
public:
2022-12-02 14:43:06 +08:00
using Ptr = std::shared_ptr<TitleMeta>;
2019-06-28 17:25:53 +08:00
TitleMeta(float dur_sec = 0,
size_t fileSize = 0,
const std::map<std::string, std::string> &header = std::map<std::string, std::string>());
2019-06-28 17:25:53 +08:00
CodecId getCodecId() const override{
return CodecInvalid;
}
};
class VideoMeta : public Metadata{
2019-06-28 17:25:53 +08:00
public:
2022-12-02 14:43:06 +08:00
using Ptr = std::shared_ptr<VideoMeta>;
2019-06-28 17:25:53 +08:00
2020-12-05 12:22:17 +08:00
VideoMeta(const VideoTrack::Ptr &video);
virtual ~VideoMeta() = default;
2019-06-28 17:25:53 +08:00
CodecId getCodecId() const override{
return _codecId;
}
private:
CodecId _codecId;
};
class AudioMeta : public Metadata{
2019-06-28 17:25:53 +08:00
public:
2022-12-02 14:43:06 +08:00
using Ptr = std::shared_ptr<AudioMeta>;
2019-06-28 17:25:53 +08:00
2020-12-05 12:22:17 +08:00
AudioMeta(const AudioTrack::Ptr &audio);
virtual ~AudioMeta() = default;
2019-06-28 17:25:53 +08:00
CodecId getCodecId() const override{
return _codecId;
}
private:
CodecId _codecId;
};
2020-04-18 22:13:11 +08:00
//根据音频track获取flags
uint8_t getAudioRtmpFlags(const Track::Ptr &track);
2019-06-28 17:25:53 +08:00
2019-03-27 18:41:52 +08:00
}//namespace mediakit
2020-05-11 22:33:10 +08:00
#endif//__rtmp_h