mirror of
				https://github.com/vdr-projects/vdr.git
				synced 2025-03-01 10:50:46 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			146 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			146 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * recorder.c: The actual DVB recorder
 | |
|  *
 | |
|  * See the main source file 'vdr.c' for copyright information and
 | |
|  * how to reach the author.
 | |
|  *
 | |
|  * $Id: recorder.c 1.10 2004/03/20 10:33:21 kls Exp $
 | |
|  */
 | |
| 
 | |
| #include <stdarg.h>
 | |
| #include <stdio.h>
 | |
| #include <unistd.h>
 | |
| #include "recorder.h"
 | |
| 
 | |
| // The size of the array used to buffer video data:
 | |
| // (must be larger than MINVIDEODATA - see remux.h)
 | |
| #define VIDEOBUFSIZE  MEGABYTE(5)
 | |
| 
 | |
| // The maximum time we wait before assuming that a recorded video data stream
 | |
| // is broken:
 | |
| #define MAXBROKENTIMEOUT 30 // seconds
 | |
| 
 | |
| #define MINFREEDISKSPACE    (512) // MB
 | |
| #define DISKCHECKINTERVAL   100 // seconds
 | |
| 
 | |
| cRecorder::cRecorder(const char *FileName, int Ca, int Priority, int VPid, int APid1, int APid2, int DPid1, int DPid2)
 | |
| :cReceiver(Ca, Priority, Setup.RecordDolbyDigital ? 5 : 3, VPid, APid1, APid2, DPid1, DPid2)
 | |
| ,cThread("recording")
 | |
| {
 | |
|   ringBuffer = NULL;
 | |
|   remux = NULL;
 | |
|   fileName = NULL;
 | |
|   index = NULL;
 | |
|   pictureType = NO_PICTURE;
 | |
|   fileSize = 0;
 | |
|   active = false;
 | |
|   lastDiskSpaceCheck = time(NULL);
 | |
| 
 | |
|   // Make sure the disk is up and running:
 | |
| 
 | |
|   SpinUpDisk(FileName);
 | |
| 
 | |
|   ringBuffer = new cRingBufferLinear(VIDEOBUFSIZE, TS_SIZE * 2, true);
 | |
|   remux = new cRemux(VPid, APid1, APid2, DPid1, DPid2, true);
 | |
|   fileName = new cFileName(FileName, true);
 | |
|   recordFile = fileName->Open();
 | |
|   if (recordFile < 0)
 | |
|      return;
 | |
|   // Create the index file:
 | |
|   index = new cIndexFile(FileName, true);
 | |
|   if (!index)
 | |
|      esyslog("ERROR: can't allocate index");
 | |
|      // let's continue without index, so we'll at least have the recording
 | |
| }
 | |
| 
 | |
| cRecorder::~cRecorder()
 | |
| {
 | |
|   Detach();
 | |
|   delete index;
 | |
|   delete fileName;
 | |
|   delete remux;
 | |
|   delete ringBuffer;
 | |
| }
 | |
| 
 | |
| void cRecorder::Activate(bool On)
 | |
| {
 | |
|   if (On) {
 | |
|      if (recordFile >= 0)
 | |
|         Start();
 | |
|      }
 | |
|   else if (active) {
 | |
|      active = false;
 | |
|      Cancel(3);
 | |
|      }
 | |
| }
 | |
| 
 | |
| bool cRecorder::RunningLowOnDiskSpace(void)
 | |
| {
 | |
|   if (time(NULL) > lastDiskSpaceCheck + DISKCHECKINTERVAL) {
 | |
|      int Free = FreeDiskSpaceMB(fileName->Name());
 | |
|      lastDiskSpaceCheck = time(NULL);
 | |
|      if (Free < MINFREEDISKSPACE) {
 | |
|         dsyslog("low disk space (%d MB, limit is %d MB)", Free, MINFREEDISKSPACE);
 | |
|         return true;
 | |
|         }
 | |
|      }
 | |
|   return false;
 | |
| }
 | |
| 
 | |
| bool cRecorder::NextFile(void)
 | |
| {
 | |
|   if (recordFile >= 0 && pictureType == I_FRAME) { // every file shall start with an I_FRAME
 | |
|      if (fileSize > MEGABYTE(Setup.MaxVideoFileSize) || RunningLowOnDiskSpace()) {
 | |
|         recordFile = fileName->NextFile();
 | |
|         fileSize = 0;
 | |
|         }
 | |
|      }
 | |
|   return recordFile >= 0;
 | |
| }
 | |
| 
 | |
| void cRecorder::Receive(uchar *Data, int Length)
 | |
| {
 | |
|   int p = ringBuffer->Put(Data, Length);
 | |
|   if (p != Length && active)
 | |
|      ringBuffer->ReportOverflow(Length - p);
 | |
| }
 | |
| 
 | |
| void cRecorder::Action(void)
 | |
| {
 | |
|   time_t t = time(NULL);
 | |
|   active = true;
 | |
|   while (active) {
 | |
|         int r;
 | |
|         const uchar *b = ringBuffer->Get(r);
 | |
|         if (b) {
 | |
|            int Count = r, Result;
 | |
|            uchar *p = remux->Process(b, Count, Result, &pictureType);
 | |
|            ringBuffer->Del(Count);
 | |
|            if (p) {
 | |
|               //XXX+ active??? see old version (Busy)
 | |
|               if (!active && pictureType == I_FRAME) // finish the recording before the next 'I' frame
 | |
|                  break;
 | |
|               if (NextFile()) {
 | |
|                  if (index && pictureType != NO_PICTURE)
 | |
|                     index->Write(pictureType, fileName->Number(), fileSize);
 | |
|                  if (safe_write(recordFile, p, Result) < 0) {
 | |
|                     LOG_ERROR_STR(fileName->Name());
 | |
|                     break;
 | |
|                     }
 | |
|                  fileSize += Result;
 | |
|                  }
 | |
|               else
 | |
|                  break;
 | |
|               }
 | |
|            t = time(NULL);
 | |
|            }
 | |
|         else if (time(NULL) - t > MAXBROKENTIMEOUT) {
 | |
|            esyslog("ERROR: video data stream broken");
 | |
|            cThread::EmergencyExit(true);
 | |
|            t = time(NULL);
 | |
|            }
 | |
|         else
 | |
|            usleep(1); // this keeps the CPU load low
 | |
|         }
 | |
| }
 |