Mp4Maker.cpp 14.3 KB
Newer Older
xiongziliang committed
1
/*
xiongziliang committed
2 3
 * MIT License
 *
xiongziliang committed
4
 * Copyright (c) 2016-2019 xiongziliang <771730766@qq.com>
xiongziliang committed
5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25
 *
 * This file is part of ZLMediaKit(https://github.com/xiongziliang/ZLMediaKit).
 *
 * 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.
 */
xzl committed
26

27
#ifdef ENABLE_MP4RECORD
28
#include <ctime>
xzl committed
29
#include <sys/stat.h>
xiongziliang committed
30
#include "Common/config.h"
xzl committed
31
#include "Mp4Maker.h"
xiongzilaing committed
32 33 34
#include "MediaRecorder.h"
#include "Util/File.h"
#include "Util/mini.h"
xiongziliang committed
35
#include "Util/util.h"
xzl committed
36
#include "Util/NoticeCenter.h"
37
#include "Extension/H264.h"
zqsong committed
38
#include "Extension/H265.h"
39
#include "Extension/AAC.h"
40
#include "Thread/WorkThreadPool.h"
41

zqsong committed
42 43 44 45

#include "mov-buffer.h"
#include "mov-format.h"

46 47 48 49 50 51


using namespace toolkit;

namespace mediakit {

zqsong committed
52 53 54 55 56 57 58 59
#if defined(_WIN32) || defined(_WIN64)
#define fseek64 _fseeki64
#define ftell64 _ftelli64
#else
#define fseek64 fseek
#define ftell64 ftell
#endif

60
static int movfileRead(void* fp, void* data, uint64_t bytes)
zqsong committed
61 62 63 64 65 66
{
    if (bytes == fread(data, 1, bytes, (FILE*)fp))
        return 0;
	return 0 != ferror((FILE*)fp) ? ferror((FILE*)fp) : -1 /*EOF*/;
}

67
static int movfileWrite(void* fp, const void* data, uint64_t bytes)
zqsong committed
68 69 70 71
{
	return bytes == fwrite(data, 1, bytes, (FILE*)fp) ? 0 : ferror((FILE*)fp);
}

72
static int movfileSeek(void* fp, uint64_t offset)
zqsong committed
73 74 75 76
{
	return fseek64((FILE*)fp, offset, SEEK_SET);
}

77
static uint64_t movfileTell(void* fp)
zqsong committed
78 79 80 81
{
	return ftell64((FILE*)fp);
}

82
const struct mov_buffer_t* movfileBuffer(void)
zqsong committed
83 84
{
	static struct mov_buffer_t s_io = {
85 86 87 88
		movfileRead,
		movfileWrite,
		movfileSeek,
		movfileTell,
zqsong committed
89 90 91 92
	};
	return &s_io;
}

xzl committed
93 94 95 96

string timeStr(const char *fmt) {
	std::tm tm_snapshot;
	auto time = ::time(NULL);
xiongziliang committed
97 98
#if defined(_WIN32)
	localtime_s(&tm_snapshot, &time); // thread-safe
xzl committed
99 100
#else
	localtime_r(&time, &tm_snapshot); // POSIX
xiongziliang committed
101
#endif
xzl committed
102 103 104 105 106 107 108 109
	const size_t size = 1024;
	char buffer[size];
	auto success = std::strftime(buffer, size, fmt, &tm_snapshot);
	if (0 == success)
		return string(fmt);
	return buffer;
}

110 111 112
Mp4Maker::Mp4Maker(const string& strPath,
				   const string &strVhost,
				   const string &strApp,
113
				   const string &strStreamId) {
xzl committed
114
	DebugL << strPath;
115
	_strPath = strPath;
xzl committed
116 117

	/////record 业务逻辑//////
118 119 120 121
	_info.strAppName = strApp;
	_info.strStreamId = strStreamId;
	_info.strVhost = strVhost;
	_info.strFolder = strPath;
122 123 124 125 126

	memset(&_movH265info, 0, sizeof(_movH265info));
	_movH265info.videoTrack = -1;	
	_movH265info.audioTrack = -1;
	
xzl committed
127 128 129 130 131 132
	//----record 业务逻辑----//
}
Mp4Maker::~Mp4Maker() {
	closeFile();
}

133
void Mp4Maker::inputH264(void *pData, uint32_t ui32Length, uint32_t ui32TimeStamp){
134 135 136
	auto iType = H264_TYPE(((uint8_t*)pData)[4]);
	
	if (H264Frame::NAL_B_P <= iType && iType <= H264Frame::NAL_IDR){	
137

138 139 140 141 142 143 144 145 146 147 148 149 150 151 152
		int64_t iTimeInc = (int64_t)ui32TimeStamp - (int64_t)_ui32LastVideoTime;
		iTimeInc = MAX(0,MIN(iTimeInc,500));
		if( (iTimeInc == 0 ||  iTimeInc == 500) && H264Frame::NAL_IDR != iType){
			WarnL << "abnormal time stamp increment:" << ui32TimeStamp << " " << _ui32LastVideoTime;
		}		
	
		if ( _strLastVideo.size() ){
			//如果出现SPS PPS黏连的帧,那么在这里先处理
			inputH264_l((char *) _strLastVideo.data(), _strLastVideo.size(), _ui32LastVideoTime); 	
			_strLastVideo = "";			
		}				
		inputH264_l((char *) pData, ui32Length, ui32TimeStamp); 	
		_ui32LastVideoTime = ui32TimeStamp;
	}else{
		//SPS PPS 进入等待组合为一帧
153
		_strLastVideo.append((char *)pData,ui32Length);
154
		_ui32LastVideoTime = ui32TimeStamp;
xzl committed
155 156
	}
}
zqsong committed
157 158

void Mp4Maker::inputH265(void *pData, uint32_t ui32Length, uint32_t ui32TimeStamp){
159

zqsong committed
160
	auto iType = H265_TYPE(((uint8_t*)pData)[4]);
161 162 163 164 165 166 167 168 169 170 171
	if (iType <= H265Frame::NAL_IDR_W_RADL ){
		
		int64_t iTimeInc = (int64_t)ui32TimeStamp - (int64_t)_ui32LastVideoTime;
		iTimeInc = MAX(0,MIN(iTimeInc,500));
		if((iTimeInc == 0 ||  iTimeInc == 500) && H265Frame::NAL_IDR_W_RADL != iType){
			WarnL << "abnormal time stamp increment:" << ui32TimeStamp << " " << _ui32LastVideoTime;
		}	
		
		if ( _strLastVideo.size() ){
			//如果出现SPS PPS VPS黏连的帧,那么在这里先处理
			inputH265_l((char *) _strLastVideo.data(), _strLastVideo.size(), _ui32LastVideoTime); 	
zqsong committed
172
			_strLastVideo = "";
173 174 175
		}
		inputH265_l((char *) pData, ui32Length, ui32TimeStamp);	
		_ui32LastVideoTime = ui32TimeStamp;
zqsong committed
176 177 178 179 180 181
	}else{
		_strLastVideo.append((char *)pData,ui32Length);
		_ui32LastVideoTime = ui32TimeStamp;
	}
}

182 183 184 185 186 187 188 189
void Mp4Maker::inputAAC(void *pData, uint32_t ui32Length, uint32_t ui32TimeStamp){	
	int64_t iTimeInc = (int64_t)ui32TimeStamp - (int64_t)_ui32LastAudioTime;
	iTimeInc = MAX(0,MIN(iTimeInc,500));
	if(iTimeInc == 0 ||  iTimeInc == 500){
		WarnL << "abnormal time stamp increment:" << ui32TimeStamp << " " << _ui32LastAudioTime;
	}		
	inputAAC_l((char *) pData, ui32Length, ui32TimeStamp);
	_ui32LastAudioTime = ui32TimeStamp;
xzl committed
190 191
}

192
void Mp4Maker::inputH264_l(void *pData, uint32_t ui32Length, uint32_t ui32TimeStamp) {
193
    GET_CONFIG(uint32_t,recordSec,Record::kFileSecond);
xiongziliang committed
194
	auto iType =  H264_TYPE(((uint8_t*)pData)[4]);
195 196
	int32_t compositionTime;
	if( iType >= H264Frame::NAL_IDR && (_movH265info.pMov == NULL || _ticker.elapsedTime() > recordSec * 1000)){
xzl committed
197 198
		//在I帧率处新建MP4文件
		//如果文件未创建或者文件超过10分钟则创建新文件
199 200 201
		//每一个录制的MP4文件时间戳都要从0开始		
		_startPts = ui32TimeStamp;		
		createFile();	
xzl committed
202
	}
203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242

	char *pNualData = (char *)pData;
	if (_movH265info.pMov!=NULL){
		int vcl;		
		if (_movH265info.videoTrack < 0){
			//解析解析SPS PPS,未添加track的时候执行
			int n = h264_annexbtomp4(&_movH265info.avc, pData, ui32Length, _sBbuffer, sizeof(_sBbuffer), &vcl);
			if (_movH265info.avc.nb_sps < 1 || _movH265info.avc.nb_pps < 1){
				return; // waiting for sps/pps
			}
			
			uint8_t sExtraData[64 * 1024];
			int extraDataSize = mpeg4_avc_decoder_configuration_record_save(&_movH265info.avc, sExtraData, sizeof(sExtraData));
			if (extraDataSize <= 0){
				// invalid HVCC
				return;
			}

			// TODO: waiting for key frame ???
			_movH265info.videoTrack = mov_writer_add_video(_movH265info.pMov,
															MOV_OBJECT_H264, 
															_movH265info.width, 
															_movH265info.height, 
															sExtraData, 
															extraDataSize);
			return; 		
		}
		if ( iType <= H264Frame::NAL_IDR ){
			uint8_t *ptr = (uint8_t*)pData;
			ptr[0] = (uint8_t)((ui32Length-4 >> 24) & 0xFF);
			ptr[1] = (uint8_t)((ui32Length-4 >> 16) & 0xFF);
			ptr[2] = (uint8_t)((ui32Length-4 >> 8) & 0xFF);
			ptr[3] = (uint8_t)((ui32Length-4 >> 0) & 0xFF);
			uint32_t ui32Pts = ui32TimeStamp < _movH265info.startPts ? 0 : ui32TimeStamp-_movH265info.startPts;
			mov_writer_write(_movH265info.pMov, 
								_movH265info.videoTrack, pData, ui32Length, 
								ui32Pts, 
								ui32Pts, 
								iType == H264Frame::NAL_IDR ? MOV_AV_FLAG_KEYFREAME : 0);
		}
xzl committed
243
	}
244

xzl committed
245 246
}

zqsong committed
247
void Mp4Maker::inputH265_l(void *pData, uint32_t ui32Length, uint32_t ui32TimeStamp) {
248
    GET_CONFIG(uint32_t,recordSec,Record::kFileSecond);
zqsong committed
249 250 251 252 253
	
	int32_t compositionTime;
	auto iType =  H265_TYPE(((uint8_t*)pData)[4]);
	if( iType >= H265Frame::NAL_IDR_W_RADL && (_movH265info.pMov == NULL || _ticker.elapsedTime() > recordSec * 1000)){
		_h265Record = 1;
254 255 256 257 258
		//每一个录制的MP4文件时间戳都要从0开始
		_startPts = ui32TimeStamp;
		
		//在I帧率处新建MP4文件
		//如果文件未创建或者文件超过最长时间则创建新文件
259 260
		createFile();
	}
zqsong committed
261 262

	char *pNualData = (char *)pData;
263
	if (_movH265info.pMov!=NULL){
zqsong committed
264 265
		int vcl;
		if (_movH265info.videoTrack < 0){
266 267
			//解析解析VPS SPS PPS,未添加track的时候执行
			int n = h265_annexbtomp4(&_movH265info.hevc, pData, ui32Length, _sBbuffer, sizeof(_sBbuffer), &vcl);
zqsong committed
268 269 270 271 272 273 274 275 276 277 278 279
			if (_movH265info.hevc.numOfArrays < 1){
				return; // waiting for vps/sps/pps
			}
			
			uint8_t sExtraData[64 * 1024];
			int extraDataSize = mpeg4_hevc_decoder_configuration_record_save(&_movH265info.hevc, sExtraData, sizeof(sExtraData));
			if (extraDataSize <= 0){
				// invalid HVCC
				return;
			}
			// TODO: waiting for key frame ???
			_movH265info.videoTrack = mov_writer_add_video(_movH265info.pMov, MOV_OBJECT_HEVC, _movH265info.width, _movH265info.height, sExtraData, extraDataSize);
280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298

			return;			
		}
		if ( iType <= H265Frame::NAL_IDR_W_RADL )
		{
			uint8_t *ptr = (uint8_t*)pData;
			ptr[0] = (uint8_t)((ui32Length-4 >> 24) & 0xFF);
			ptr[1] = (uint8_t)((ui32Length-4 >> 16) & 0xFF);
			ptr[2] = (uint8_t)((ui32Length-4 >> 8) & 0xFF);
			ptr[3] = (uint8_t)((ui32Length-4 >> 0) & 0xFF);

			uint32_t ui32Pts = ui32TimeStamp < _movH265info.startPts ? 0 : ui32TimeStamp-_movH265info.startPts;
			mov_writer_write(_movH265info.pMov,
							_movH265info.videoTrack,
							pData, 
							ui32Length, 
							ui32Pts, 
							ui32Pts, 
							iType == H265Frame::NAL_IDR_W_RADL ? MOV_AV_FLAG_KEYFREAME : 0);
zqsong committed
299 300 301 302 303
		}
	}

}

304
void Mp4Maker::inputAAC_l(void *pData, uint32_t ui32Length, uint32_t ui32TimeStamp) {
zqsong committed
305
    GET_CONFIG(uint32_t,recordSec,Record::kFileSecond);
306 307 308 309
	if (!_haveVideo && (_movH265info.pMov == NULL || _ticker.elapsedTime() > recordSec * 1000)) {			
		_startPts = ui32TimeStamp;
		createFile();			
	}
zqsong committed
310

311 312 313 314
	if (-1 != _movH265info.audioTrack && _movH265info.pMov != NULL){	

		uint32_t ui32Pts = ui32TimeStamp < _movH265info.startPts ? 0 : ui32TimeStamp-_movH265info.startPts;
		mov_writer_write(_movH265info.pMov, _movH265info.audioTrack,  (uint8_t*)pData, ui32Length, ui32Pts, ui32Pts, 0);		
315
	}
316
	
xzl committed
317 318 319 320 321 322
}

void Mp4Maker::createFile() {
	closeFile();
	auto strDate = timeStr("%Y-%m-%d");
	auto strTime = timeStr("%H-%M-%S");
323 324
	auto strFileTmp = _strPath + strDate + "/." + strTime + ".mp4";
	auto strFile =	_strPath + strDate + "/" + strTime + ".mp4";
xzl committed
325 326

	/////record 业务逻辑//////
327 328 329
	_info.ui64StartedTime = ::time(NULL);
	_info.strFileName = strTime + ".mp4";
	_info.strFilePath = strFile;
330

331
    GET_CONFIG(string,appName,Record::kAppName);
332 333 334 335 336 337

    _info.strUrl = appName + "/"
                   + _info.strAppName + "/"
                   + _info.strStreamId + "/"
                   + strDate + "/"
                   + strTime + ".mp4";
338

xzl committed
339 340
	//----record 业务逻辑----//

xiongziliang committed
341
#if !defined(_WIN32)
xzl committed
342
	File::createfile_path(strFileTmp.data(), S_IRWXO | S_IRWXG | S_IRWXU);
xiongziliang committed
343 344 345
#else
	File::createfile_path(strFileTmp.data(), 0);
#endif
zqsong committed
346

347 348 349 350 351 352 353 354
	memset(&_movH265info, 0, sizeof(_movH265info));
	_movH265info.videoTrack = -1;	
	_movH265info.audioTrack = -1;
	_movH265info.width = 0;
	_movH265info.height = 0;
	_movH265info.pFile = fopen(strFileTmp.data(), "wb+");
	_movH265info.startPts = _startPts;
	_movH265info.pMov  = mov_writer_create(movfileBuffer(), _movH265info.pFile, 0/*MOV_FLAG_FASTSTART*/);
zqsong committed
355

356 357 358
	_strFileTmp = strFileTmp;
	_strFile = strFile;
	_ticker.resetTime();
xiongziliang committed
359

zqsong committed
360 361 362 363 364
	if ( _h265Record ){
		auto videoTrack = dynamic_pointer_cast<H265Track>(getTrack(TrackVideo));
		if(videoTrack){
			_movH265info.width = videoTrack->getVideoWidth();
			_movH265info.height = videoTrack->getVideoHeight();
365
		}
366
	}else{
zqsong committed
367 368
		auto videoTrack = dynamic_pointer_cast<H264Track>(getTrack(TrackVideo));
		if(videoTrack){
369 370 371 372 373
//			auto &sps = videoTrack->getSps();
//			auto &pps = videoTrack->getPps();
			_movH265info.width = videoTrack->getVideoWidth();
			_movH265info.height = videoTrack->getVideoHeight();
		}		
374
	}
375 376
	if ( _movH265info.width <=0 || _movH265info.height <= 0 )
		WarnL << "分辨率获取失败,MP4录制异常";
377 378 379 380

	auto audioTrack = dynamic_pointer_cast<AACTrack>(getTrack(TrackAudio));
	if(audioTrack){
		_audioSampleRate = audioTrack->getAudioSampleRate();
zqsong committed
381
		_audioChannel = audioTrack->getAudioChannel();
382 383 384 385
		uint8_t extra_data[64 * 1024];		
		_movH265info.audioTrack = mov_writer_add_audio(_movH265info.pMov, MOV_OBJECT_AAC, _audioChannel, 16, _audioSampleRate, audioTrack->getAacCfg().data(), 2);
		if (-1 == _movH265info.audioTrack) 
			WarnL << "添加音频通道失败:" << strFileTmp;
386
	}
xzl committed
387 388
}

389 390 391 392
void Mp4Maker::asyncClose() {
	auto strFileTmp = _strFileTmp;
	auto strFile = _strFile;
	auto info = _info;
zqsong committed
393
	
394 395 396
	FILE *pFile = _movH265info.pFile;
	mov_writer_t*  hMp4 = _movH265info.pMov;
	WorkThreadPool::Instance().getExecutor()->async([hMp4,strFileTmp,strFile,info,pFile]() {
397 398 399
		//获取文件录制时间,放在MP4Close之前是为了忽略MP4Close执行时间
		const_cast<Mp4Info&>(info).ui64TimeLen = ::time(NULL) - info.ui64StartedTime;
		//MP4Close非常耗时,所以要放在后台线程执行
400 401
		if (hMp4 != NULL ){
			mov_writer_destroy(hMp4);			
zqsong committed
402
			fclose(pFile);
403
			DebugL << "fclose end";
zqsong committed
404
		}
405

406 407 408 409 410 411 412 413 414 415 416
		//临时文件名改成正式文件名,防止mp4未完成时被访问
		rename(strFileTmp.data(),strFile.data());
		//获取文件大小
		struct stat fileData;
		stat(strFile.data(), &fileData);
		const_cast<Mp4Info&>(info).ui64FileSize = fileData.st_size;
		/////record 业务逻辑//////
		NoticeCenter::Instance().emitEvent(Broadcast::kBroadcastRecordMP4,info);
	});
}

xzl committed
417
void Mp4Maker::closeFile() {
418 419 420 421 422 423 424
	if (_movH265info.pMov != NULL) {	
		asyncClose();
		memset(&_movH265info, 0, sizeof(_movH265info));
		_movH265info.pMov = NULL;
		_movH265info.videoTrack = -1;	
		_movH265info.audioTrack = -1;
	}	
xzl committed
425 426
}

427 428 429
void Mp4Maker::onTrackFrame(const Frame::Ptr &frame) {
	switch (frame->getCodecId()){
		case CodecH264:{
430 431
			//需要带00 00 00 01,方便在mov_writer_write的时候字节修改这4Byte为长度信息
			inputH264(frame->data() , frame->size(),frame->pts());
432 433 434
		}
			break;
		case CodecAAC:{
435 436
			//不需要ADTS头
			inputAAC(frame->data() + frame->prefixSize(), frame->size() - frame->prefixSize(),frame->pts());
437 438
		}
			break;
zqsong committed
439
		case CodecH265:{
440 441
			//需要带00 00 00 01,方便在mov_writer_write的时候字节修改这4Byte为长度信息
			inputH265(frame->data() , frame->size(),frame->pts());
zqsong committed
442 443
		}
			break;
444 445 446 447 448 449 450 451 452
		default:
			break;
	}
}

void Mp4Maker::onAllTrackReady() {
	_haveVideo = getTrack(TrackVideo).operator bool();
}

xiongziliang committed
453
} /* namespace mediakit */
xzl committed
454 455


456
#endif //ENABLE_MP4RECORD