/* * @DEC_COPYRIGHT@ */ /* * HISTORY * $Log: slib_api.c,v $ * Revision 1.1.6.35 1996/12/13 18:19:04 Hans_Graves * Update Audio and Video timestamp correctly. * [1996/12/13 18:09:02 Hans_Graves] * * Revision 1.1.6.34 1996/12/12 20:54:44 Hans_Graves * Timestamp fixes after seek to Key frames. * [1996/12/12 20:52:06 Hans_Graves] * * Revision 1.1.6.33 1996/12/10 19:46:02 Hans_Graves * Fix floating division error when audio only. * [1996/12/10 19:45:03 Hans_Graves] * * Revision 1.1.6.32 1996/12/10 19:21:55 Hans_Graves * Made calculate video positions more accurate using slibFrameToTime100(). * [1996/12/10 19:16:20 Hans_Graves] * * Revision 1.1.6.31 1996/12/05 20:10:15 Hans_Graves * Add gradual increase or decrease of framerates according to timestamps. * [1996/12/05 20:06:57 Hans_Graves] * * Revision 1.1.6.30 1996/12/04 22:34:28 Hans_Graves * Put limit on data used by Sv/SaDecompressBegin(). * [1996/12/04 22:14:33 Hans_Graves] * * Revision 1.1.6.29 1996/12/03 23:15:13 Hans_Graves * MME-1498: Made seeks with PERCENT100 more accurate * [1996/12/03 23:10:43 Hans_Graves] * * Revision 1.1.6.28 1996/12/03 00:08:31 Hans_Graves * Handling of End Of Sequence points. Added PERCENT100 support. * [1996/12/03 00:05:59 Hans_Graves] * * Revision 1.1.6.27 1996/11/21 23:34:21 Hans_Graves * Handle MPEG B frames better when seeking. * [1996/11/21 23:28:18 Hans_Graves] * * Revision 1.1.6.26 1996/11/20 02:15:09 Hans_Graves * Added SEEK_AHEAD. Removed old code. * [1996/11/20 02:10:43 Hans_Graves] * * Revision 1.1.6.25 1996/11/18 23:07:21 Hans_Graves * Remove MaxVideoLength usage. * [1996/11/18 22:55:56 Hans_Graves] * * Make use of presentation timestamps. Make seeking time-based. * [1996/11/18 22:47:30 Hans_Graves] * * Revision 1.1.6.24 1996/11/14 21:49:26 Hans_Graves * AC3 buffering fixes. * [1996/11/14 21:43:20 Hans_Graves] * * Revision 1.1.6.23 1996/11/13 16:10:54 Hans_Graves * AC3 recognition of byte reversed streams in slibGetDataFormat(). * [1996/11/13 16:03:14 Hans_Graves] * * Revision 1.1.6.22 1996/11/11 18:21:03 Hans_Graves * More AC3 support changes. * [1996/11/11 17:59:01 Hans_Graves] * * Revision 1.1.6.21 1996/11/08 21:51:02 Hans_Graves * Added AC3 support. Better seperation of stream types. * [1996/11/08 21:27:57 Hans_Graves] * * Revision 1.1.6.20 1996/10/31 00:08:51 Hans_Graves * Fix skipping data after RESET with MPEG video only streams. * [1996/10/31 00:07:08 Hans_Graves] * * Revision 1.1.6.19 1996/10/28 23:16:42 Hans_Graves * MME-0145?, Fix artifacts when using SlibReadData() at a new position. Jump to first GOP. * [1996/10/28 23:13:01 Hans_Graves] * * Revision 1.1.6.18 1996/10/28 17:32:28 Hans_Graves * MME-1402, 1431, 1435: Timestamp related changes. * [1996/10/28 17:22:58 Hans_Graves] * * Revision 1.1.6.17 1996/10/17 00:23:32 Hans_Graves * Fix buffer problems after SlibQueryData() calls. * [1996/10/17 00:19:05 Hans_Graves] * * Revision 1.1.6.16 1996/10/15 17:34:09 Hans_Graves * Added MPEG-2 Program Stream support. * [1996/10/15 17:30:26 Hans_Graves] * * Revision 1.1.6.15 1996/10/12 17:18:51 Hans_Graves * Fixed some seeking problems. Moved render code to slib_render.c * [1996/10/12 17:00:49 Hans_Graves] * * Revision 1.1.6.14 1996/10/03 19:14:21 Hans_Graves * Added Presentation and Decoding timestamp support. * [1996/10/03 19:10:35 Hans_Graves] * * Revision 1.1.6.13 1996/09/29 22:19:37 Hans_Graves * Added stride support. Added SlibQueryData(). * [1996/09/29 21:29:44 Hans_Graves] * * Revision 1.1.6.12 1996/09/25 19:16:44 Hans_Graves * Added DECOMPRESS_QUERY. Fix up support for YUY2. * [1996/09/25 19:00:45 Hans_Graves] * * Revision 1.1.6.11 1996/09/23 18:04:03 Hans_Graves * Added stats support. Scaleing and negative height fixes. * [1996/09/23 17:59:31 Hans_Graves] * * Revision 1.1.6.10 1996/09/18 23:46:32 Hans_Graves * Seek fixes. Added SlibReadData() and SlibAddBufferEx(). * [1996/09/18 22:04:57 Hans_Graves] * * Revision 1.1.6.9 1996/08/09 20:51:42 Hans_Graves * Fix handle arg for SlibRegisterVideoBuffer() * [1996/08/09 20:10:11 Hans_Graves] * * Revision 1.1.6.8 1996/07/19 02:11:11 Hans_Graves * Added SlibRegisterVideoBuffer. Added YUV422i to RGB 16 rendering. * [1996/07/19 02:01:11 Hans_Graves] * * Revision 1.1.6.7 1996/06/03 21:41:12 Hans_Graves * Fix file seeking. Always seeked to position 0. * [1996/06/03 21:40:44 Hans_Graves] * * Revision 1.1.6.6 1996/05/24 22:21:44 Hans_Graves * Merge MME-1221. Last SlibReadAudio() returned EndOfStream even if data read. * [1996/05/24 20:58:42 Hans_Graves] * * Revision 1.1.6.5 1996/05/23 18:46:35 Hans_Graves * Seperate global audio and video SInfo variables, to help multi-threaded apps * [1996/05/23 18:35:14 Hans_Graves] * * Revision 1.1.6.4 1996/05/23 18:16:31 Hans_Graves * Added more YUV Conversions. MPEG audio buffering fix. * [1996/05/23 18:16:11 Hans_Graves] * * Revision 1.1.6.3 1996/05/10 21:17:00 Hans_Graves * Added callback support. Also fill entire buffers when calling SlibReadAudio() * [1996/05/10 20:26:08 Hans_Graves] * * Revision 1.1.6.2 1996/05/07 19:56:16 Hans_Graves * Added SlibOpen() and SlibAddBuffer() framework. Added HUFF_SUPPORT. * [1996/05/07 17:20:12 Hans_Graves] * * Revision 1.1.4.16 1996/05/02 17:10:33 Hans_Graves * Be more specific about checking for MPEG-2 Systems file type. Fixes MME-01234 * [1996/05/02 17:04:44 Hans_Graves] * * Revision 1.1.4.15 1996/04/24 22:33:44 Hans_Graves * MPEG encoding bitrate fixups. * [1996/04/24 22:27:09 Hans_Graves] * * Revision 1.1.4.14 1996/04/23 21:22:31 Hans_Graves * Added description for SlibErrorSettingNotEqual * [1996/04/23 21:16:09 Hans_Graves] * * Revision 1.1.4.13 1996/04/22 15:04:51 Hans_Graves * Fix bad frame counts and seeking under NT caused by int overflows * [1996/04/22 15:02:26 Hans_Graves] * * Revision 1.1.4.12 1996/04/19 21:52:22 Hans_Graves * MPEG 1 Systems writing enhancements * [1996/04/19 21:47:48 Hans_Graves] * * Revision 1.1.4.11 1996/04/15 14:18:37 Hans_Graves * Handle any audio buffer size during encoding. * [1996/04/15 14:16:11 Hans_Graves] * * Revision 1.1.4.10 1996/04/12 19:25:20 Hans_Graves * Add MPEG2_VIDEO type to Commit * [1996/04/12 19:24:19 Hans_Graves] * * Revision 1.1.4.9 1996/04/10 21:47:41 Hans_Graves * Fix in SlibIsEnd(). * [1996/04/10 21:39:37 Hans_Graves] * * Revision 1.1.4.8 1996/04/09 16:04:42 Hans_Graves * Remove NT warnings * [1996/04/09 14:42:48 Hans_Graves] * * Revision 1.1.4.7 1996/04/04 23:35:07 Hans_Graves * Format conversion cleanup * [1996/04/04 23:16:20 Hans_Graves] * * Revision 1.1.4.6 1996/04/01 19:07:52 Hans_Graves * And some error checking * [1996/04/01 19:04:33 Hans_Graves] * * Revision 1.1.4.5 1996/04/01 16:23:12 Hans_Graves * NT porting * [1996/04/01 16:15:54 Hans_Graves] * * Revision 1.1.4.4 1996/03/29 22:21:30 Hans_Graves * Added MPEG/JPEG/H261_SUPPORT ifdefs * [1996/03/29 21:56:55 Hans_Graves] * * Added MPEG-I Systems encoding support * [1996/03/27 21:55:54 Hans_Graves] * * Revision 1.1.4.3 1996/03/12 16:15:45 Hans_Graves * Added seperate streams to SlibIsEnd() * [1996/03/12 15:56:28 Hans_Graves] * * Revision 1.1.4.2 1996/03/08 18:46:42 Hans_Graves * YUV conversions moved to slibRenderFrame() * [1996/03/08 18:14:47 Hans_Graves] * * Revision 1.1.2.18 1996/02/22 23:30:24 Hans_Graves * Update FPS on seeks * [1996/02/22 23:29:27 Hans_Graves] * * Revision 1.1.2.17 1996/02/22 22:23:56 Hans_Graves * Update frame numbers with timecode more often * [1996/02/22 22:23:07 Hans_Graves] * * Revision 1.1.2.16 1996/02/21 22:52:43 Hans_Graves * Fixed MPEG 2 systems stuff * [1996/02/21 22:50:55 Hans_Graves] * * Revision 1.1.2.15 1996/02/19 20:09:28 Hans_Graves * Debugging message clean-up * [1996/02/19 20:08:31 Hans_Graves] * * Revision 1.1.2.14 1996/02/19 18:03:54 Hans_Graves * Fixed a number of MPEG related bugs * [1996/02/19 17:57:36 Hans_Graves] * * Revision 1.1.2.13 1996/02/13 18:47:46 Hans_Graves * Fix some Seek related bugs * [1996/02/13 18:40:36 Hans_Graves] * * Revision 1.1.2.12 1996/02/07 23:23:54 Hans_Graves * Added SEEK_EXACT. Fixed most frame counting problems. * [1996/02/07 23:20:29 Hans_Graves] * * Revision 1.1.2.11 1996/02/06 22:54:05 Hans_Graves * Seek fix-ups. More accurate MPEG frame counts. * [1996/02/06 22:44:56 Hans_Graves] * * Revision 1.1.2.10 1996/02/02 17:36:02 Hans_Graves * Enhanced audio info. Cleaned up API * [1996/02/02 17:29:44 Hans_Graves] * * Revision 1.1.2.9 1996/01/30 22:23:08 Hans_Graves * Added AVI YUV support * [1996/01/30 22:21:38 Hans_Graves] * * Revision 1.1.2.8 1996/01/15 16:26:27 Hans_Graves * Removed debuging message * [1996/01/15 16:02:47 Hans_Graves] * * Added MPEG 1 Audio compression and SlibWriteAudio() * [1996/01/15 15:45:46 Hans_Graves] * * Revision 1.1.2.7 1996/01/11 16:17:29 Hans_Graves * Added MPEG II Systems decode support * [1996/01/11 16:12:33 Hans_Graves] * * Revision 1.1.2.6 1996/01/08 16:41:31 Hans_Graves * Added MPEG II decoding support * [1996/01/08 15:53:02 Hans_Graves] * * Revision 1.1.2.5 1995/12/08 20:01:20 Hans_Graves * Fixed SlibSetParam(). Added H.261 compression support. * [1995/12/08 19:53:52 Hans_Graves] * * Revision 1.1.2.4 1995/12/07 19:31:36 Hans_Graves * Added JPEG Decoding and MPEG encoding support * [1995/12/07 18:30:10 Hans_Graves] * * Revision 1.1.2.3 1995/11/09 23:14:05 Hans_Graves * Added MPEG audio decompression * [1995/11/09 23:08:33 Hans_Graves] * * Revision 1.1.2.2 1995/11/06 18:47:52 Hans_Graves * First time under SLIB * [1995/11/06 18:36:01 Hans_Graves] * * $EndLog$ */ /***************************************************************************** ** Copyright (c) Digital Equipment Corporation, 1995 ** ** ** ** All Rights Reserved. Unpublished rights reserved under the copyright ** ** laws of the United States. ** ** ** ** The software contained on this media is proprietary to and embodies ** ** the confidential technology of Digital Equipment Corporation. ** ** Possession, use, duplication or dissemination of the software and ** ** media is authorized only pursuant to a valid written license from ** ** Digital Equipment Corporation. ** ** ** ** RESTRICTED RIGHTS LEGEND Use, duplication, or disclosure by the U.S. ** ** Government is subject to restrictions as set forth in Subparagraph ** ** (c)(1)(ii) of DFARS 252.227-7013, or in FAR 52.227-19, as applicable. ** ******************************************************************************/ /* #define _SLIBDEBUG_ */ #include #include #ifdef WIN32 #include #endif #ifdef _SHM_ #include /* shared memory */ #endif #define SLIB_INTERNAL #include "slib.h" #include "SC_err.h" /* #include "SC_convert.h" */ #include "mpeg.h" #include "ac3.h" #include "avi.h" #ifdef _SLIBDEBUG_ #include #include "sc_debug.h" #define _DEBUG_ 1 /* detailed debuging statements */ #define _VERBOSE_ 1 /* show progress */ #define _VERIFY_ 1 /* verify correct operation */ #define _WARN_ 1 /* warnings about strange behavior */ #define _SEEK_ 1 /* seek, frame counts/timecode info: 2=more detail */ #define _CALLBACK_ 0 /* callback debugging */ #define _DUMP_ 0 /* dump data in hex format */ #define _TIMECODE_ 1 /* debug timecodes */ #endif static SlibStatus_t slibOpen(SlibHandle_t *handle, SlibMode_t smode, SlibType_t *stype); /* ** Lists */ static SlibList_t _listTypes [] = { SLIB_TYPE_MPEG1_VIDEO, "MPEG1_VIDEO", "MPEG-1 Video Stream",0,0, SLIB_TYPE_MPEG1_AUDIO, "MPEG1_AUDIO", "MPEG-1 Audio Stream",0,0, SLIB_TYPE_MPEG2_VIDEO, "MPEG2_VIDEO", "MPEG-2 Video Stream",0,0, SLIB_TYPE_MPEG2_AUDIO, "MPEG2_AUDIO", "MPEG-2 Audio Stream",0,0, SLIB_TYPE_AC3_AUDIO, "AC3", "Dolby Digital(AC-3) Stream",0,0, SLIB_TYPE_MPEG_SYSTEMS, "MPEG_SYSTEMS", "MPEG Systems Stream",0,0, SLIB_TYPE_MPEG_SYSTEMS_MPEG2, "MPEG_SYSTEMS_MPEG2", "MPEG Systems (MPEG-2)",0,0, SLIB_TYPE_MPEG_TRANSPORT,"MPEG_TRANSPORT", "MPEG Transport Stream",0,0, SLIB_TYPE_MPEG_PROGRAM, "MPEG_PROGRAM", "MPEG Program Stream",0,0, SLIB_TYPE_H261, "H261", "H.261 Video Stream",0,0, SLIB_TYPE_RTP_H261, "RTP_H261", "RTP (H.261) Stream",0,0, SLIB_TYPE_H263, "H263", "H.263 Video Stream",0,0, SLIB_TYPE_RTP_H263, "RTP_H263", "RTP (H.263) Stream",0,0, SLIB_TYPE_RIFF, "RIFF", "RIFF File Format",0,0, SLIB_TYPE_AVI, "AVI", "AVI File Format",0,0, SLIB_TYPE_PCM_WAVE, "PCM_WAVE", "WAVE (PCM) File Format",0,0, SLIB_TYPE_JPEG_AVI, "JPEG_AVI", "AVI (JPEG) Stream",0,0, SLIB_TYPE_MJPG_AVI, "MJPG_AVI", "AVI (MJPG) Stream",0,0, SLIB_TYPE_YUV_AVI, "YUV_AVI", "AVI (YUV) File Format",0,0, SLIB_TYPE_JFIF, "JFIF", "JPEG (JFIF) Stream",0,0, SLIB_TYPE_JPEG_QUICKTIME,"JPEG_QUICKTIME","JPEG (Quicktime) Stream",0,0, SLIB_TYPE_JPEG, "JPEG", "JPEG Stream",0,0, SLIB_TYPE_MJPG, "MJPG", "MJPG Stream",0,0, SLIB_TYPE_YUV, "YUV", "YUV Data",0,0, SLIB_TYPE_RGB, "RGB", "RGB Data",0,0, SLIB_TYPE_PCM, "PCM", "PCM Audio",0,0, SLIB_TYPE_SLIB, "SLIB", "SLIB Stream",0,0, SLIB_TYPE_SHUFF, "SHUFF", "SLIB Huffman Stream",0,0, SLIB_TYPE_G723, "G723", "G.723 Audio Stream",0,0, SLIB_TYPE_RASTER, "RASTER","Sun Raster",0,0, SLIB_TYPE_BMP, "BMP", "Windows Bitmap",0,0, 0, NULL, "End of List",0,0 }; static SlibList_t _listCompressTypes [] = { #ifdef MPEG_SUPPORT SLIB_TYPE_MPEG1_VIDEO, "MPEG1_VIDEO", "MPEG-1 Video Stream",0,0, SLIB_TYPE_MPEG1_AUDIO, "MPEG1_AUDIO", "MPEG-1 Audio Stream",0,0, SLIB_TYPE_MPEG2_VIDEO, "MPEG2_VIDEO", "MPEG-2 Video Stream",0,0, SLIB_TYPE_MPEG_SYSTEMS, "MPEG_SYSTEMS", "MPEG Systems Stream",0,0, SLIB_TYPE_MPEG_SYSTEMS_MPEG2, "MPEG2_SYSTEMS", "MPEG Systems (MPEG-2)",0,0, #endif /* MPEG_SUPPORT */ #ifdef H261_SUPPORT SLIB_TYPE_H261, "H261", "H.261 Video Stream",0,0, #endif /* H261_SUPPORT */ #ifdef H263_SUPPORT SLIB_TYPE_H263, "H263", "H.263 Video Stream",0,0, #endif /* H263_SUPPORT */ #ifdef HUFF_SUPPORT SLIB_TYPE_SHUFF, "SHUFF", "SLIB Huffman Stream",0,0, #endif /* HUFF_SUPPORT */ #ifdef G723_SUPPORT SLIB_TYPE_G723, "G723", "G.723 Audio Stream",0,0, #endif /* G723_SUPPORT */ 0, NULL, "End of List",0,0 }; static SlibList_t _listDecompressTypes [] = { #ifdef MPEG_SUPPORT SLIB_TYPE_MPEG1_VIDEO, "MPEG1_VIDEO", "MPEG-1 Video Stream",0,0, SLIB_TYPE_MPEG1_AUDIO, "MPEG1_AUDIO", "MPEG-1 Audio Stream",0,0, SLIB_TYPE_MPEG2_VIDEO, "MPEG2_VIDEO", "MPEG-2 Video Stream",0,0, SLIB_TYPE_MPEG_SYSTEMS, "MPEG_SYSTEMS", "MPEG Systems Stream",0,0, SLIB_TYPE_MPEG_SYSTEMS_MPEG2, "MPEG2_SYSTEMS", "MPEG Systems (MPEG-2)",0,0, SLIB_TYPE_MPEG_TRANSPORT, "MPEG_TRANSPORT", "MPEG Transport Stream",0,0, SLIB_TYPE_MPEG_PROGRAM, "MPEG_PROGRAM", "MPEG Program Stream",0,0, #endif /* MPEG_SUPPORT */ #ifdef AC3_SUPPORT SLIB_TYPE_AC3_AUDIO, "AC3", "Dolby Digital(AC-3) Stream",0,0, #endif /* AC3_SUPPORT */ #ifdef H261_SUPPORT SLIB_TYPE_H261, "H261", "H.261 Video Stream",0,0, SLIB_TYPE_RTP_H261, "RTP_H261", "RTP (H.261) Stream",0,0, #endif /* H261_SUPPORT */ #ifdef H263_SUPPORT SLIB_TYPE_H263, "H263", "H.263 Video Stream",0,0, SLIB_TYPE_RTP_H263, "RTP_H263", "RTP (H.263) Stream",0,0, #endif /* H261_SUPPORT */ #ifdef JPEG_SUPPORT SLIB_TYPE_JPEG_AVI, "JPEG_AVI", "AVI (JPEG) Stream",0,0, SLIB_TYPE_MJPG_AVI, "MJPG_AVI", "AVI (MJPG) Stream",0,0, #endif /* JPEG_SUPPORT */ SLIB_TYPE_RIFF, "RIFF", "RIFF File Format",0,0, SLIB_TYPE_AVI, "AVI", "AVI File Format",0,0, SLIB_TYPE_PCM_WAVE, "PCM_WAVE", "WAVE (PCM) File Format",0,0, SLIB_TYPE_YUV_AVI, "YUV_AVI", "AVI (YUV) File Format",0,0, SLIB_TYPE_RASTER, "RASTER","Sun Raster",0,0, SLIB_TYPE_BMP, "BMP", "Windows Bitmap",0,0, #ifdef HUFF_SUPPORT SLIB_TYPE_SHUFF, "SHUFF", "SLIB Huffman Stream",0,0, #endif /* HUFF_SUPPORT */ #ifdef G723_SUPPORT SLIB_TYPE_G723, "G723", "G.723 Audio Stream",0,0, #endif /* G723_SUPPORT */ 0, NULL, "End of List",0,0 }; static SlibList_t _listErrors[] = { SlibErrorNone, "SlibErrorNone", "No Error",0,0, SlibErrorInternal, "SlibErrorInternal", "Internal SLIB error",0,0, SlibErrorMemory, "SlibErrorMemory", "Unable to allocated memory",0,0, SlibErrorBadArgument, "SlibErrorBadArgument", "Invalid argument to function",0,0, SlibErrorBadHandle, "SlibErrorBadHandle", "Invalid SLIB handle",0,0, SlibErrorBadMode, "SlibErrorBadMode", "Invalid SLIB mode",0,0, SlibErrorUnsupportedFormat, "SlibErrorUnsupportedFormat", "Unsupported format",0,0, SlibErrorReading, "SlibErrorReading", "Error reading from file",0,0, SlibErrorWriting, "SlibErrorWriting", "Error writing to file",0,0, SlibErrorBufSize, "SlibErrorBufSize", "Buffer size is too small",0,0, SlibErrorEndOfStream, "SlibErrorEndOfStream", "End of data stream",0,0, SlibErrorForwardOnly, "SlibErrorForwardOnly", "The decompressor can work only forward",0,0, SlibErrorUnsupportedParam, "SlibErrorUnsupportedParam", "The parameter is invalid or unsupported",0,0, SlibErrorImageSize, "SlibErrorImageSize", "Invalid image height and/or width size",0,0, SlibErrorSettingNotEqual, "SlibErrorSettingNotEqual", "The exact parameter setting was not used",0,0, SlibErrorInit, "SlibErrorInit", "Initializing CODEC failed",0,0, SlibErrorFileSize, "SlibErrorFileSize", "Error in file size",0,0, SlibErrorBadPosition, "SlibErrorBadPosition", "Error in seek position",0,0, SlibErrorBadUnit, "SlibErrorBadUnit", "Error in seek units",0,0, SlibErrorNoData, "SlibErrorNoData", "No data available",0,0, 0, NULL, "End of List",0,0 }; SlibList_t *SlibFindEnumEntry(SlibList_t *list, int enumval) { if (!list) return(NULL); while (list->Name) { if (list->Enum==enumval) return(list); list++; } return(NULL); } char *SlibGetErrorText(SlibStatus_t status) { SlibList_t *entry=SlibFindEnumEntry(_listErrors, status); if (entry) return(entry->Desc); else return(NULL); } int VCompressCallback(SvHandle_t Svh, SvCallbackInfo_t *CB, SvPictureInfo_t *pinfo) { int status; SlibInfo_t *Info=(SlibInfo_t *)CB->UserData; _SlibDebug(_CALLBACK_, printf("VCompressCallback()\n") ); switch (CB->Message) { case CB_END_BUFFERS: _SlibDebug(_CALLBACK_, printf("VCompressCallback received CB_END_BUFFER message\n") ); if (CB->DataType==CB_DATA_COMPRESSED) { CB->DataSize=Info->CompBufSize; CB->Data=SlibAllocBuffer(CB->DataSize); status=SvAddBuffer(Svh, CB); _SlibDebug(_WARN_ && status!=NoErrors, printf("SvAddBuffer() %s\n", ScGetErrorStr(status)) ); } break; case CB_RELEASE_BUFFER: _SlibDebug(_CALLBACK_, printf("VCompressCallback received CB_RELEASE_BUFFER message\n")); if (CB->DataType==CB_DATA_COMPRESSED && CB->Data && CB->DataUsed) { SlibPin_t *dstpin=slibGetPin(Info, SLIB_DATA_VIDEO); if (dstpin) { slibAddBufferToPin(dstpin, CB->Data, CB->DataUsed, Info->VideoPTimeCode); Info->VideoPTimeCode=SLIB_TIME_NONE; if (!slibCommitBuffers(Info, FALSE)) CB->Action = CB_ACTION_END; break; } } if (CB->Data) SlibFreeBuffer(CB->Data); break; case CB_FRAME_START: _SlibDebug(_CALLBACK_||_TIMECODE_, printf("VCompress CB_FRAME_START: TimeCode=%ld TemporalRef=%d\n", pinfo->TimeCode, pinfo->TemporalRef) ); Info->VideoPTimeCode=pinfo->TimeCode; #if 0 if (pinfo->Type==SV_I_PICTURE || pinfo->Type==SV_P_PICTURE) { if (!SlibTimeIsValid(Info->LastVideoDTimeCode)) Info->VideoDTimeCode=-1000/(long)Info->FramesPerSec; else Info->VideoDTimeCode=Info->LastVideoDTimeCode; Info->LastVideoDTimeCode=pinfo->TimeCode; _SlibDebug(_CALLBACK_||_TIMECODE_, printf("CB_FRAME_START: LastVideoDTimeCode=%ld VideoDTimeCode=%ld\n", Info->LastVideoDTimeCode, Info->VideoDTimeCode)); } else Info->VideoDTimeCode=-1; #endif break; } CB->Action = CB_ACTION_CONTINUE; return(NoErrors); } int VDecompressCallback(SvHandle_t Svh, SvCallbackInfo_t *CB, SvPictureInfo_t *PictInfo) { int status; unsigned dword size; SlibInfo_t *Info=(SlibInfo_t *)CB->UserData; _SlibDebug(_CALLBACK_, printf("VDecompressCallback()\n") ); switch (CB->Message) { case CB_SEQ_END: /* reset presentation timestamps at end-of-sequence */ _SlibDebug(_CALLBACK_ || _TIMECODE_, printf("VDecompressCallback received CB_SEQ_END message\n") ); Info->VideoPTimeCode=SLIB_TIME_NONE; Info->AudioPTimeCode=SLIB_TIME_NONE; Info->VideoTimeStamp=SLIB_TIME_NONE; Info->AudioTimeStamp=SLIB_TIME_NONE; break; case CB_END_BUFFERS: _SlibDebug(_CALLBACK_, printf("VDecompressCallback received CB_END_BUFFER message\n") ); if (CB->DataType==CB_DATA_COMPRESSED) { SlibTime_t ptimestamp, timediff; slibSetMaxInput(Info, 1500*1024); /* set limit for input data */ CB->Data = SlibGetBuffer(Info, SLIB_DATA_VIDEO, &size, &ptimestamp); slibSetMaxInput(Info, 0); /* clear limit */ CB->DataSize = size; if (SlibTimeIsValid(Info->AudioPTimeCode)) { timediff=ptimestamp-Info->AudioPTimeCode; if (timediff>6000) { /* Make sure a NEW audio time is not way out of * sync with video time. * This can happen after an End of Sequence. */ /* assign audio time to video time */ Info->VideoPTimeCode=SLIB_TIME_NONE; Info->VideoPTimeBase=Info->AudioPTimeBase; ptimestamp=Info->AudioPTimeCode; } } if (SlibTimeIsValid(ptimestamp) && ptimestamp>Info->VideoPTimeCode) { SlibTime_t lasttime=Info->VideoPTimeCode; Info->VideoPTimeCode=ptimestamp; _SlibDebug(_CALLBACK_||_TIMECODE_, printf("VideoPTimeCode=%ld\n", Info->VideoPTimeCode) ); ptimestamp-=Info->VideoPTimeBase; timediff=ptimestamp-Info->VideoTimeStamp; if (SlibTimeIsInValid(lasttime) || SlibTimeIsInValid(Info->VideoTimeStamp)) { _SlibDebug(_TIMECODE_, printf("Updating VideoTimeStamp none->%ld\n", ptimestamp) ); Info->VideoTimeStamp=ptimestamp; Info->AvgVideoTimeDiff=0; Info->VarVideoTimeDiff=0; } else /* see if times are far off */ { SlibTime_t lastavg=Info->AvgVideoTimeDiff; Info->AvgVideoTimeDiff=(lastavg*14+timediff)/15; Info->VarVideoTimeDiff=(Info->VarVideoTimeDiff*3+ lastavg-Info->AvgVideoTimeDiff)/4; _SlibDebug(_CALLBACK_||_TIMECODE_, printf("Video timediff: Cur=%ld Avg=%ld Var=%ld\n", timediff, Info->AvgVideoTimeDiff, Info->VarVideoTimeDiff)); if (Info->VarVideoTimeDiff==0) { _SlibDebug(_TIMECODE_, printf("Updating VideoTimeStamp %ld->%ld (diff=%ld)\n", Info->VideoTimeStamp, ptimestamp, ptimestamp-Info->VideoTimeStamp) ); Info->VideoTimeStamp=ptimestamp; Info->AvgVideoTimeDiff=0; } else if (Info->AvgVideoTimeDiff>=100 || Info->AvgVideoTimeDiff<=-100) { /* calculated time and timestamps are too far off */ float fps=Info->FramesPerSec; if (Info->VarVideoTimeDiff>1 && fps>=15.5F) fps-=0.25F; /* playing too fast, slow frame rate */ else if (Info->VarVideoTimeDiff<-1 && fps<=59.0F) fps+=0.25F; /* playing too slow, speed up frame rate */ _SlibDebug(_WARN_ || _CALLBACK_||_TIMECODE_, printf("Updating fps from %.4f -> %.4f\n", Info->FramesPerSec, fps) ); Info->FramesPerSec=fps; Info->VideoFrameDuration=slibFrameToTime100(Info, 1); Info->VideoTimeStamp=ptimestamp; Info->AvgVideoTimeDiff=0; } } Info->VideoFramesProcessed=0; /* reset frames processed */ } if (CB->DataSize>0) { _SlibDebug(_DUMP_, SlibPin_t *pin=slibGetPin(Info, SLIB_DATA_VIDEO); printf("VDecompressCallback() Adding buffer of length %d\n", CB->DataSize); ScDumpChar(CB->Data, (int)CB->DataSize, (int)pin->Offset-CB->DataSize)); CB->DataType = CB_DATA_COMPRESSED; _SlibDebug(_CALLBACK_, printf("VDecompressCallback() Adding buffer of length %d\n", CB->DataSize) ); status = SvAddBuffer(Svh, CB); } else { _SlibDebug(_WARN_ || _CALLBACK_, printf("VDecompressCallback() got no data\n") ); CB->Action = CB_ACTION_END; return(NoErrors); } } break; case CB_RELEASE_BUFFER: _SlibDebug(_CALLBACK_, printf("VDecompressCallback received CB_RELEASE_BUFFER message\n")); if (CB->DataType==CB_DATA_COMPRESSED && CB->Data) SlibFreeBuffer(CB->Data); break; case CB_PROCESSING: _SlibDebug(_CALLBACK_, printf("VDecompressCallback received CB_PROCESSING message\n") ); break; case CB_CODEC_DONE: _SlibDebug(_CALLBACK_, printf("VDecompressCallback received CB_CODEC_DONE message\n") ); break; } CB->Action = CB_ACTION_CONTINUE; return(NoErrors); } int ACompressCallback(SaHandle_t Sah, SaCallbackInfo_t *CB, SaInfo_t *SaInfo) { int status; SlibInfo_t *Info=(SlibInfo_t *)CB->UserData; _SlibDebug(_CALLBACK_, printf("ACompressCallback()\n") ); CB->Action = CB_ACTION_CONTINUE; switch (CB->Message) { case CB_END_BUFFERS: _SlibDebug(_CALLBACK_, printf("ACompressCallback received CB_END_BUFFER message\n") ); if (CB->DataType==CB_DATA_COMPRESSED) { CB->DataSize=Info->CompBufSize; CB->Data=SlibAllocBuffer(CB->DataSize); _SlibDebug(_CALLBACK_, printf("ACompressCallback() Adding buffer of length %d\n", CB->DataSize) ); status=SaAddBuffer(Sah, CB); _SlibDebug(_WARN_ && status!=NoErrors, printf("SaAddBuffer() %s\n", ScGetErrorStr(status)) ); } break; case CB_RELEASE_BUFFER: _SlibDebug(_CALLBACK_, printf("ACompressCallback received CB_RELEASE_BUFFER message\n")); if (CB->DataType==CB_DATA_COMPRESSED && CB->Data && CB->DataUsed) { SlibPin_t *dstpin=slibGetPin(Info, SLIB_DATA_AUDIO); if (dstpin) { slibAddBufferToPin(dstpin, CB->Data, CB->DataUsed, Info->AudioPTimeCode); Info->AudioPTimeCode=SLIB_TIME_NONE; if (!slibCommitBuffers(Info, FALSE)) CB->Action = CB_ACTION_END; } } else if (CB->Data) SlibFreeBuffer(CB->Data); break; case CB_FRAME_START: _SlibDebug(_CALLBACK_||_TIMECODE_, printf("ACompress CB_FRAME_START: TimeStamp=%ld Frame=%d\n", CB->TimeStamp, Info->VideoFramesProcessed ) ); if (SlibTimeIsInValid(Info->AudioPTimeCode)) { Info->AudioPTimeCode=CB->TimeStamp; _SlibDebug(_TIMECODE_, printf("AudioPTimeCode=%ld\n", Info->AudioPTimeCode) ); _SlibDebug(_WARN_ && (Info->AudioTimeStamp-CB->TimeStamp>400 || CB->TimeStamp-Info->AudioTimeStamp>400), printf("Bad Audio Time: AudioPTimeCode=%ld AudioTimestamp=%ld\n", Info->AudioPTimeCode, Info->AudioTimeStamp) ); } break; } return(NoErrors); } int ADecompressCallback(SvHandle_t Sah, SaCallbackInfo_t *CB, SaInfo_t *SaInfo) { int status; unsigned dword size; SlibInfo_t *Info=(SlibInfo_t *)CB->UserData; _SlibDebug(_DEBUG_, printf("ADecompressCallback()\n") ); switch (CB->Message) { case CB_END_BUFFERS: _SlibDebug(_CALLBACK_, printf("ADecompressCallback() CB_END_BUFFER\n") ); if (CB->DataType==CB_DATA_COMPRESSED) { SlibTime_t ptimestamp, timediff; slibSetMaxInput(Info, 2000*1024); /* set limit for input data */ CB->Data = SlibGetBuffer(Info, SLIB_DATA_AUDIO, &size, &ptimestamp); slibSetMaxInput(Info, 0); /* clear limit */ CB->DataSize = size; if (SlibTimeIsValid(ptimestamp)) { Info->AudioPTimeCode=ptimestamp; _SlibDebug(_CALLBACK_||_TIMECODE_, printf("AudioPTimeCode=%ld\n", Info->AudioPTimeCode) ); ptimestamp-=Info->AudioPTimeBase; timediff=ptimestamp-Info->AudioTimeStamp; if (SlibTimeIsInValid(Info->AudioTimeStamp)) Info->AudioTimeStamp=ptimestamp; else if (timediff<-300 || timediff>300) /* time is far off */ { _SlibDebug(_WARN_||_TIMECODE_, printf("Updating AudioTimeStamp %ld->%ld (diff=%ld)\n", Info->AudioTimeStamp, ptimestamp,timediff) ); Info->AudioTimeStamp=ptimestamp; if (SlibTimeIsValid(Info->VideoTimeStamp)) { /* Make sure a NEW audio time is not way out of * sync with video time. * This can happen after an End of Sequence. */ timediff=ptimestamp-Info->VideoTimeStamp; if (timediff<-6000) { /* assign audio time to video time */ Info->VideoPTimeCode=SLIB_TIME_NONE; Info->VideoPTimeBase=Info->AudioPTimeBase; Info->VideoTimeStamp=ptimestamp; } } } } if (CB->Data) { if (CB->DataSize>0) { CB->DataType = CB_DATA_COMPRESSED; _SlibDebug(_CALLBACK_, printf("ADecompressCallback() Adding buffer of length %d\n", CB->DataSize) ); status = SaAddBuffer(Sah, CB); } else SlibFreeBuffer(CB->Data); } else { _SlibDebug(_WARN_ || _CALLBACK_, printf("ADecompressCallback() got no data\n") ); CB->Action = CB_ACTION_END; return(NoErrors); } } break; case CB_RELEASE_BUFFER: _SlibDebug(_CALLBACK_, printf("ADecompressCallback() CB_RELEASE_BUFFER\n")); if (CB->DataType==CB_DATA_COMPRESSED && CB->Data) SlibFreeBuffer(CB->Data); break; case CB_PROCESSING: _SlibDebug(_CALLBACK_, printf("ADecompressCallback() CB_PROCESSING\n") ); break; case CB_CODEC_DONE: _SlibDebug(_CALLBACK_, printf("ADecompressCallback() CB_CODEC_DONE\n") ); break; } CB->Action = CB_ACTION_CONTINUE; return(NoErrors); } static void slibInitInfo(SlibInfo_t *Info) { _SlibDebug(_DEBUG_, printf("slibInitInfo()\n") ); Info->Type = SLIB_TYPE_UNKNOWN; Info->Mode = SLIB_MODE_NONE; Info->Svh = NULL; Info->Sah = NULL; Info->Sch = NULL; Info->NeedAccuracy = FALSE; Info->TotalBitRate = 0; Info->MuxBitRate = 0; Info->SystemTimeStamp = 0; /* Audio parameters */ Info->AudioStreams = 0; Info->SamplesPerSec = 0; Info->BitsPerSample = 0; Info->Channels = 0; Info->AudioBitRate = 0; Info->AudioMainStream = 0; Info->AudioType = SLIB_TYPE_UNKNOWN; /* Video parameters */ Info->VideoStreams = 0; Info->Width = 0; Info->Height = 0; Info->Stride = 0; Info->VideoBitRate = 0; Info->FramesPerSec = 0.0F; Info->ImageSize = 0; Info->AudioPID = -1; Info->VideoPID = -1; Info->VideoMainStream = 0; Info->VideoType = SLIB_TYPE_UNKNOWN; /* Data Exchange */ Info->Offset = 0; Info->Pins = NULL; Info->PinCount = 0; Info->IOError = FALSE; Info->MaxBytesInput = 0; Info->BytesProcessed = 0; /* stream dependent stuff */ Info->VideoLength = 0; Info->VideoLengthKnown = FALSE; Info->VideoTimeStamp = SLIB_TIME_NONE; Info->VideoFrameDuration = 0; Info->AudioLength = 0; Info->AudioLengthKnown = FALSE; Info->AudioTimeStamp = SLIB_TIME_NONE; Info->LastAudioTimeStamp = SLIB_TIME_NONE; Info->KeySpacing = 0; Info->SubKeySpacing = 0; Info->VideoPTimeBase = SLIB_TIME_NONE; Info->VideoPTimeCode = SLIB_TIME_NONE; Info->VideoDTimeCode = SLIB_TIME_NONE; Info->LastAudioPTimeCode = SLIB_TIME_NONE; Info->LastVideoPTimeCode = SLIB_TIME_NONE; Info->LastVideoDTimeCode = SLIB_TIME_NONE; Info->AvgVideoTimeDiff = 0; Info->VarVideoTimeDiff = 0; Info->AudioPTimeBase = SLIB_TIME_NONE; Info->AudioPTimeCode = SLIB_TIME_NONE; Info->AudioDTimeCode = SLIB_TIME_NONE; Info->VideoFramesProcessed=0; /* Encoding info */ Info->HeaderProcessed = FALSE; Info->PacketCount = 0; Info->BytesSincePack = 0; /* Miscellaneous */ Info->SlibCB = NULL; Info->SlibCBUserData = NULL; Info->Fd = -1; Info->FileSize = 0; Info->FileBufSize = 50*1024; Info->CompBufSize = 2*1024; Info->PacketSize = 512; Info->AudioFormat = NULL; Info->VideoFormat = NULL; Info->CompAudioFormat = NULL; Info->CompVideoFormat = NULL; Info->CodecVideoFormat = NULL; Info->VideoCodecState = SLIB_CODEC_STATE_NONE; Info->AudioCodecState = SLIB_CODEC_STATE_NONE; Info->Imagebuf = NULL; Info->IntImagebuf = NULL; Info->IntImageSize = 0; Info->CodecImagebuf = NULL; Info->CodecImageSize = 0; Info->Multibuf = NULL; Info->MultibufSize = 0; Info->Audiobuf = NULL; Info->AudiobufSize = 0; Info->AudiobufUsed = 0; Info->OverflowSize = 1500*1024; Info->VBVbufSize = 0; Info->stats = NULL; Info->dbg = NULL; } /* ** Name: slibGetDataFormat ** Purpose: Find out the type of some multmedia data. */ static SlibType_t slibGetDataFormat(unsigned char *buf, int size, dword *headerstart, dword *headersize) { dword i, count; unsigned char *bufptr; if (headersize) *headersize=0; if (size<4 || !buf) return(SLIB_TYPE_UNKNOWN); /* ** H261 video stream file */ if ((buf[0] == 0x00) && (buf[1] == 0x01) && (buf[2] & 0xF0)==0x00) return(SLIB_TYPE_H261); /* ** H263 video stream file */ if ((buf[0] == 0x00) && (buf[1] == 0x00) && (buf[2] == 0x80) && (buf[3] & 0xF8)==0x00) return(SLIB_TYPE_H263); /* ** JFIF file (ffd8 = Start-Of-Image marker) */ if (buf[0] == 0xff && buf[1] == 0xd8) return(SLIB_TYPE_JFIF); /* ** QUICKTIME JPEG file (4 ignored bytes, "mdat", ff, d8, ff) */ if ((strncmp(&buf[4], "mdat", 4) == 0 ) && (buf[8] == 0xff) && (buf[9] == 0xd8) && (buf[10] == 0xff)) return(SLIB_TYPE_JPEG_QUICKTIME); /* ** AVI RIFF file */ if ( strncmp(buf, "RIFF", 4) == 0 ) { if (strncmp(&buf[8], "WAVE",4) == 0) return(SLIB_TYPE_PCM_WAVE); else if (strncmp(&buf[8], "AVI ",4) == 0) return(SLIB_TYPE_AVI); else return(SLIB_TYPE_RIFF); } /* ** BMP file */ if (buf[0] == 'B' && buf[1]=='M') return(SLIB_TYPE_BMP); /* ** Dolby AC-3 stream */ if ((buf[0]==0x77 && buf[1] == 0x0B) || /* may be byte reversed */ (buf[0]==0x0B && buf[1] == 0x77)) return(SLIB_TYPE_AC3_AUDIO); /* ** Sun Raster file */ if ((buf[0]==0x59 && buf[1] == 0xA6) || /* may be byte reversed */ (buf[0]==0x6A && buf[1] == 0x95)) return(SLIB_TYPE_RASTER); /* ** SLIB file */ if ((buf[0] == 'S') && (buf[1] == 'L') && (buf[2] == 'I') && (buf[3] == 'B')) { if ((buf[4] == 'H') && (buf[5] == 'U') && /* SLIB Huffman Stream */ (buf[6] == 'F') && (buf[7] == 'F')) return(SLIB_TYPE_SHUFF); else return(SLIB_TYPE_SLIB); } /* ** MPEG II Transport Stream */ if (buf[0] == MPEG_TSYNC_CODE && (buf[3]&0x30)!=0) /* adaptation field value is not reserved */ return(SLIB_TYPE_MPEG_TRANSPORT); if (buf[0] == MPEG_TSYNC_CODE && buf[1] == 0x1F && buf[2]==0xFF) /* NULL PID */ return(SLIB_TYPE_MPEG_TRANSPORT); /* search for mpeg startcode 000001 */ bufptr=buf; for (i=4, count=size; i11 && (bufptr[11]&0x02)) /* load_intra_quantizer_matrixe */ { *headersize+=64; if (count>75 && bufptr[64+11]&0x01) /* load_non_intra_quantizer_matrix */ *headersize+=64; } else if (count>11 && (bufptr[11]&0x01)) /* load_non_intra_quant_matrix */ *headersize+=64; } return(SLIB_TYPE_MPEG1_VIDEO); } /* ** MPEG I Systems file */ if ((bufptr[0] == 0x00) && (bufptr[1] == 0x00) && (bufptr[2] == 0x01) && (bufptr[3] == 0xba) && ((bufptr[4]&0xF0) == 0x20)) return(SLIB_TYPE_MPEG_SYSTEMS); /* ** MPEG II Program Stream */ if ((bufptr[0] == 0x00) && (bufptr[1] == 0x00) && (bufptr[2] == 0x01) && (bufptr[3] == 0xba) && ((bufptr[4]&0xC0) == 0x40)) return(SLIB_TYPE_MPEG_PROGRAM); /* ** H263 video stream file */ /* search for H.263 picture startcode 000000000000000100000 */ for (bufptr=buf, i=0, count=size-4; i=12 ? SLIB_TYPE_RTP_H263 : SLIB_TYPE_H263); } /* ** H261 video stream file */ /* search for H.261 picture startcode 00000000000000010000 */ for (bufptr=buf, i=0, count=size-3; i=12 ? SLIB_TYPE_RTP_H261 : SLIB_TYPE_H261); } /* ** MPEG audio stream file */ if (buf[0]==0xFF && (buf[1] & 0xF0)==0xF0) return(SLIB_TYPE_MPEG1_AUDIO); #ifdef G723_SUPPORT //Detect the RATEFLAG and VADFLAG in each frame in this //buffer. { int i,iFrameSize,iNoOfFrames; BOOL bRateFlag; //0 for High rate(6.3K 24bit), 1 for low rate(5.3K,20bit) BOOL bVADflag; // 0 for Active speech 1 for Non-speech BOOL bTypeG723 = TRUE; //Initialized to say that it's a g723 media stream if(buf[0] & 0x1) { bRateFlag = TRUE; //Low rate 5.3K iFrameSize = 20; } else { bRateFlag = FALSE; //High Rate 6.3K iFrameSize = 24; } if(buf[0] & 0x2) bVADflag =TRUE; //Non-Speech else bVADflag = FALSE; //Active-Speech iNoOfFrames = size/iFrameSize; if (iNoOfFrames>15) iNoOfFrames=15; /* just check first 15 frames */ //Leave the first frame.The first frame is used to extract // the above information.Check for this info in the remaining // frames.If it exists in all frames,the audio is G723 ,otherwise // audio type is Unknown. for(i =1; i < iNoOfFrames;i++) { //Search for RateFlag and bVADflag for each frame if(((buf[i*iFrameSize] & 0x1) == bRateFlag) && ((buf[i*iFrameSize] & 0x2) == bVADflag)) continue; //type is Unknown ,Set the flag to false and //break from the for loop bTypeG723 = FALSE; break; } if(bTypeG723) return(SLIB_TYPE_G723); } #endif /* G723_SUPPORT */ _SlibDebug(_WARN_, printf("slibGetDataFormat() Unknown file format\n") ); return(SLIB_TYPE_UNKNOWN); } SlibStatus_t SlibQueryData(void *databuf, unsigned dword databufsize, SlibQueryInfo_t *qinfo) { SlibInfo_t *Info=NULL; SlibStatus_t status; if (!databuf) return(SlibErrorBadArgument); if (databufsize==0) return(SlibErrorBufSize); qinfo->Bitrate=0; qinfo->VideoStreams=0; qinfo->Width=0; qinfo->Height=0; qinfo->VideoBitrate=0; qinfo->FramesPerSec=0.0F; qinfo->VideoLength=0; qinfo->AudioStreams=0; qinfo->SamplesPerSec=0; qinfo->BitsPerSample=0; qinfo->Channels=0; qinfo->AudioBitrate=0; qinfo->AudioLength=0; qinfo->Type = slibGetDataFormat(databuf, databufsize, &qinfo->HeaderStart, &qinfo->HeaderSize); if (qinfo->Type!=SLIB_TYPE_UNKNOWN) { if ((Info = (SlibInfo_t *)ScAlloc(sizeof(SlibInfo_t))) == NULL) return(SlibErrorMemory); slibInitInfo(Info); Info->Mode = SLIB_MODE_DECOMPRESS; Info->Type = qinfo->Type; slibAddPin(Info, SLIB_DATA_COMPRESSED, "Compressed"); status=slibManageUserBuffer(NULL, databuf, databufsize, NULL); if (status==SlibErrorNone) status=slibAddBufferToPin(slibGetPin(Info, SLIB_DATA_COMPRESSED), databuf, databufsize, SLIB_TIME_NONE); if (status!=SlibErrorNone) { SlibClose((SlibHandle_t)Info); return(status); } slibAddPin(Info, SLIB_DATA_AUDIO, "Audio"); slibAddPin(Info, SLIB_DATA_VIDEO, "Video"); SlibUpdateAudioInfo(Info); SlibUpdateVideoInfo(Info); if (Info->TotalBitRate==0) qinfo->Bitrate=Info->AudioBitRate+ ((Info->VideoBitRate>100000000)?0:Info->VideoBitRate); else qinfo->Bitrate=Info->TotalBitRate; qinfo->VideoStreams=Info->VideoStreams; qinfo->Width=Info->Width; qinfo->Height=Info->Height; qinfo->VideoBitrate=Info->VideoBitRate; qinfo->FramesPerSec=Info->FramesPerSec; qinfo->VideoLength=Info->VideoLength; qinfo->AudioStreams=Info->AudioStreams; qinfo->SamplesPerSec=Info->SamplesPerSec; qinfo->BitsPerSample=Info->BitsPerSample; qinfo->Channels=Info->Channels; qinfo->AudioBitrate=Info->AudioBitRate; qinfo->AudioLength=Info->AudioLength; SlibClose((SlibHandle_t)Info); return(SlibErrorNone); } return(SlibErrorUnsupportedFormat); } /************************** The Main Slib API ***********************/ SlibStatus_t SlibOpen(SlibHandle_t *handle, SlibMode_t smode, SlibType_t *stype, SlibMessage_t (*slibCB)(SlibHandle_t, SlibMessage_t, SlibCBParam1_t, SlibCBParam2_t, void *), void *cbuserdata) { SlibInfo_t *Info=NULL; SlibStatus_t status; _SlibDebug(_VERBOSE_,printf("SlibOpen()\n") ); if (!handle) return(SlibErrorBadHandle); *handle = NULL; if (!stype) return(SlibErrorBadArgument); if (!slibCB) return(SlibErrorBadArgument); if (smode == SLIB_MODE_COMPRESS) { if (SlibFindEnumEntry(_listCompressTypes, *stype)==NULL) return(SlibErrorUnsupportedFormat); } if ((Info = (SlibInfo_t *)ScAlloc(sizeof(SlibInfo_t))) == NULL) return(SlibErrorMemory); slibInitInfo(Info); slibAddPin(Info, SLIB_DATA_COMPRESSED, "Compressed"); Info->SlibCB = slibCB; Info->SlibCBUserData = cbuserdata; *handle=(SlibHandle_t)Info; if ((status=slibOpen(handle, smode, stype))!=SlibErrorNone) *handle = NULL; return(status); } SlibStatus_t SlibOpenSync(SlibHandle_t *handle, SlibMode_t smode, SlibType_t *stype, void *buffer, unsigned dword bufsize) { SlibInfo_t *Info=NULL; SlibStatus_t status; _SlibDebug(_VERBOSE_,printf("SlibOpenSync()\n") ); if (!handle) return(SlibErrorBadHandle); *handle = NULL; if (!stype) return(SlibErrorBadArgument); if (smode == SLIB_MODE_COMPRESS) { if (SlibFindEnumEntry(_listCompressTypes, *stype)==NULL) return(SlibErrorUnsupportedFormat); } else if (smode == SLIB_MODE_DECOMPRESS) { /* for decompression we need the first buffer to open the codecs */ if (!buffer || bufsize==0) return(SlibErrorBadArgument); } if ((Info = (SlibInfo_t *)ScAlloc(sizeof(SlibInfo_t))) == NULL) return(SlibErrorMemory); slibInitInfo(Info); Info->Mode=smode; slibAddPin(Info, SLIB_DATA_COMPRESSED, "Compressed"); if (smode == SLIB_MODE_DECOMPRESS) { status=SlibAddBuffer((SlibHandle_t *)Info, SLIB_DATA_COMPRESSED, buffer, bufsize); if (status!=SlibErrorNone) return(status); } *handle=(SlibHandle_t)Info; if ((status=slibOpen(handle, smode, stype))!=SlibErrorNone) *handle = NULL; return(status); } SlibStatus_t SlibOpenFile(SlibHandle_t *handle, SlibMode_t smode, SlibType_t *stype, char *filename) { SlibInfo_t *Info=NULL; SlibStatus_t status; _SlibDebug(_VERBOSE_,printf("SlibOpenFile()\n") ); if (!handle) return(SlibErrorBadHandle); *handle = NULL; if (!stype) return(SlibErrorBadArgument); if (!filename) return(SlibErrorBadArgument); if (smode == SLIB_MODE_COMPRESS) { if (SlibFindEnumEntry(_listCompressTypes, *stype)==NULL) return(SlibErrorUnsupportedFormat); if ((Info = (SlibInfo_t *) ScAlloc(sizeof(SlibInfo_t))) == NULL) return(SlibErrorMemory); slibInitInfo(Info); Info->Fd = ScFileOpenForWriting(filename, TRUE); if (Info->Fd<0) { ScFree(Info); return(SlibErrorWriting); } *handle=(SlibHandle_t)Info; if ((status=slibOpen(handle, smode, stype))!=SlibErrorNone) *handle = NULL; return(status); } else if (smode == SLIB_MODE_DECOMPRESS) { if ((Info = (SlibInfo_t *)ScAlloc(sizeof(SlibInfo_t))) == NULL) return(SlibErrorMemory); slibInitInfo(Info); Info->Fd = ScFileOpenForReading(filename); if (Info->Fd<0) { ScFree(Info); return(SlibErrorReading); } ScFileSize(filename, &Info->FileSize); *handle=(SlibHandle_t)Info; if ((status=slibOpen(handle, smode, stype))!=SlibErrorNone) *handle = NULL; return(status); } else return(SlibErrorBadMode); } static SlibStatus_t slibOpen(SlibHandle_t *handle, SlibMode_t smode, SlibType_t *stype) { SlibInfo_t *Info=(SlibInfo_t *)*handle; unsigned char *buf; unsigned dword size; _SlibDebug(_VERBOSE_,printf("SlibOpenFile()\n") ); if (!Info) return(SlibErrorMemory); if (!stype) return(SlibErrorBadArgument); if (Info->SlibCB) { SlibMessage_t result; _SlibDebug(_VERBOSE_, printf("slibOpen() SlibCB(SLIB_MSG_OPEN)\n") ); result=(*(Info->SlibCB))((SlibHandle_t)Info, SLIB_MSG_OPEN, (SlibCBParam1_t)0, (SlibCBParam2_t)0, (void *)Info->SlibCBUserData); } if (smode == SLIB_MODE_COMPRESS) { Info->Mode = smode; Info->Type = *stype; SlibUpdateAudioInfo(Info); SlibUpdateVideoInfo(Info); switch (Info->Type) { #ifdef MPEG_SUPPORT case SLIB_TYPE_MPEG_SYSTEMS: case SLIB_TYPE_MPEG_SYSTEMS_MPEG2: case SLIB_TYPE_MPEG1_VIDEO: Info->VideoStreams = 1; if (SvOpenCodec (Info->Type==SLIB_TYPE_MPEG_SYSTEMS_MPEG2 ? SV_MPEG2_ENCODE : SV_MPEG_ENCODE, &Info->Svh)!=SvErrorNone) { SlibClose((SlibHandle_t)Info); return(SlibErrorUnsupportedFormat); } Info->VideoCodecState=SLIB_CODEC_STATE_OPEN; slibAddPin(Info, SLIB_DATA_VIDEO, "Video"); if (Info->Type==SLIB_TYPE_MPEG1_VIDEO) break; case SLIB_TYPE_MPEG1_AUDIO: Info->AudioStreams = 1; if (SaOpenCodec (SA_MPEG_ENCODE, &Info->Sah)!=SvErrorNone) { SlibClose((SlibHandle_t)Info); return(SlibErrorUnsupportedFormat); } Info->AudioCodecState=SLIB_CODEC_STATE_OPEN; slibAddPin(Info, SLIB_DATA_AUDIO, "Audio"); break; case SLIB_TYPE_MPEG2_VIDEO: Info->VideoStreams = 1; if (SvOpenCodec (SV_MPEG2_ENCODE, &Info->Svh)!=SvErrorNone) { SlibClose((SlibHandle_t)Info); return(SlibErrorUnsupportedFormat); } Info->VideoCodecState=SLIB_CODEC_STATE_OPEN; slibAddPin(Info, SLIB_DATA_VIDEO, "Video"); break; #endif /* MPEG_SUPPORT */ #ifdef H261_SUPPORT case SLIB_TYPE_H261: Info->VideoStreams = 1; if (SvOpenCodec (SV_H261_ENCODE, &Info->Svh)!=SvErrorNone) { SlibClose((SlibHandle_t)Info); return(SlibErrorUnsupportedFormat); } Info->VideoCodecState=SLIB_CODEC_STATE_OPEN; slibAddPin(Info, SLIB_DATA_VIDEO, "Video"); break; #endif /* H261_SUPPORT */ #ifdef H263_SUPPORT case SLIB_TYPE_H263: Info->VideoStreams = 1; if (SvOpenCodec (SV_H263_ENCODE, &Info->Svh)!=SvErrorNone) { SlibClose((SlibHandle_t)Info); return(SlibErrorUnsupportedFormat); } Info->VideoCodecState=SLIB_CODEC_STATE_OPEN; slibAddPin(Info, SLIB_DATA_VIDEO, "Video"); break; #endif /* H263_SUPPORT */ #ifdef HUFF_SUPPORT case SLIB_TYPE_SHUFF: Info->VideoStreams = 1; if (SvOpenCodec (SV_HUFF_ENCODE, &Info->Svh)!=SvErrorNone) { SlibClose((SlibHandle_t)Info); return(SlibErrorUnsupportedFormat); } Info->VideoCodecState=SLIB_CODEC_STATE_OPEN; slibAddPin(Info, SLIB_DATA_VIDEO, "Video"); break; #endif /* HUFF_SUPPORT */ #ifdef G723_SUPPORT case SLIB_TYPE_G723: Info->AudioStreams = 1; if (SaOpenCodec (SA_G723_ENCODE, &Info->Sah)!=SvErrorNone) { SlibClose((SlibHandle_t)Info); return(SlibErrorUnsupportedFormat); } Info->AudioCodecState=SLIB_CODEC_STATE_OPEN; slibAddPin(Info, SLIB_DATA_AUDIO, "Audio"); break; #endif /*G723_SUPPORT*/ default: return(SlibErrorUnsupportedFormat); } slibAddPin(Info, SLIB_DATA_COMPRESSED, "Compressed"); } else if (smode == SLIB_MODE_DECOMPRESS) { Info->Mode = smode; /* ** Determine the input data type */ if (slibLoadPin(Info, SLIB_DATA_COMPRESSED)==NULL) return(SlibErrorReading); if ((buf=SlibPeekBuffer(Info, SLIB_DATA_COMPRESSED, &size, NULL))==NULL || size<=0) { /* couldn't get any compressed data */ SlibClose((SlibHandle_t)Info); return(SlibErrorReading); } Info->Type = slibGetDataFormat(buf, size, NULL, NULL); /* if we can't determine the type, use stype as the type */ if (Info->Type==SLIB_TYPE_UNKNOWN && stype) Info->Type=*stype; if (SlibFindEnumEntry(_listDecompressTypes, Info->Type)==NULL) { SlibClose((SlibHandle_t)Info); return(SlibErrorUnsupportedFormat); } slibAddPin(Info, SLIB_DATA_AUDIO, "Audio"); slibAddPin(Info, SLIB_DATA_VIDEO, "Video"); if (SlibTypeIsMPEGMux(Info->Type)) { /* need to select main streams for multiplexed streams */ Info->AudioMainStream=MPEG_AUDIO_STREAM_START; Info->VideoMainStream=MPEG_VIDEO_STREAM_START; /* private data may be needed - i.e. AC3 */ slibAddPin(Info, SLIB_DATA_PRIVATE, "Private"); } SlibUpdateAudioInfo(Info); SlibUpdateVideoInfo(Info); if (Info->AudioStreams<=0) slibRemovePin(Info, SLIB_DATA_AUDIO); if (Info->VideoStreams<=0) slibRemovePin(Info, SLIB_DATA_VIDEO); slibRemovePin(Info, SLIB_DATA_PRIVATE); /* only used in init */ if (Info->AudioBitRate && Info->VideoBitRate) { if (!Info->VideoLengthKnown) { qword ms=((qword)Info->FileSize*80L)/ (Info->AudioBitRate+Info->VideoBitRate); ms = (ms*75)/80; /* adjust for systems data */ Info->AudioLength = Info->VideoLength = (SlibTime_t)ms*100; _SlibDebug(_SEEK_||_VERBOSE_, ScDebugPrintf(Info->dbg,"slibOpen() FileSize=%ld VideoLength=%ld\n", Info->FileSize, Info->VideoLength) ); } else if (Info->VideoLengthKnown && Info->FramesPerSec) Info->AudioLength = Info->VideoLength; } if (Info->TotalBitRate==0) Info->TotalBitRate=Info->AudioBitRate + ((Info->VideoBitRate>100000000)?0:Info->VideoBitRate); _SlibDebug(_SEEK_||_VERBOSE_, ScDebugPrintf(Info->dbg,"AudioLength=%ld VideoLength=%ld %s\n", Info->AudioLength, Info->VideoLength, Info->VideoLengthKnown?"(known)":"") ); if (Info->AudioType==SLIB_TYPE_UNKNOWN && Info->VideoType==SLIB_TYPE_UNKNOWN) return(SlibErrorUnsupportedFormat); switch (Info->AudioType) { case SLIB_TYPE_UNKNOWN: break; #ifdef MPEG_SUPPORT case SLIB_TYPE_MPEG1_AUDIO: if (SaOpenCodec (SA_MPEG_DECODE, &Info->Sah)!=SvErrorNone) { SlibClose((SlibHandle_t)Info); return(SlibErrorUnsupportedFormat); } Info->AudioCodecState=SLIB_CODEC_STATE_OPEN; break; #endif /* MPEG_SUPPORT */ #ifdef GSM_SUPPORT if (SaOpenCodec (SA_GSM_DECODE, &Info->Sah)!=SvErrorNone) { SlibClose((SlibHandle_t)Info); return(SlibErrorUnsupportedFormat); } Info->AudioCodecState=SLIB_CODEC_STATE_OPEN; break; #endif /* GSM_SUPPORT */ #ifdef AC3_SUPPORT case SLIB_TYPE_AC3_AUDIO: if (SaOpenCodec (SA_AC3_DECODE, &Info->Sah)!=SvErrorNone) { SlibClose((SlibHandle_t)Info); return(SlibErrorUnsupportedFormat); } Info->AudioCodecState=SLIB_CODEC_STATE_OPEN; break; #endif /* AC3_SUPPORT */ #ifdef G723_SUPPORT case SLIB_TYPE_G723: if (SaOpenCodec (SA_G723_DECODE, &Info->Sah)!=SvErrorNone) { SlibClose((SlibHandle_t)Info); return(SlibErrorUnsupportedFormat); } Info->AudioCodecState=SLIB_CODEC_STATE_OPEN; break; #endif /* G723_SUPPORT */ } /* AudioType */ switch (Info->VideoType) { case SLIB_TYPE_UNKNOWN: break; #ifdef MPEG_SUPPORT case SLIB_TYPE_MPEG1_VIDEO: if (SvOpenCodec (SV_MPEG_DECODE, &Info->Svh)!=SvErrorNone) { SlibClose((SlibHandle_t)Info); return(SlibErrorUnsupportedFormat); } Info->VideoCodecState=SLIB_CODEC_STATE_OPEN; _SlibDebug(_DEBUG_,printf("VideoCodecState=OPEN\n")); break; case SLIB_TYPE_MPEG2_VIDEO: if (SvOpenCodec (SV_MPEG2_DECODE, &Info->Svh)!=SvErrorNone) { SlibClose((SlibHandle_t)Info); return(SlibErrorUnsupportedFormat); } Info->VideoCodecState=SLIB_CODEC_STATE_OPEN; _SlibDebug(_DEBUG_,printf("VideoCodecState=OPEN\n")); break; #endif /* MPEG_SUPPORT */ #ifdef H261_SUPPORT case SLIB_TYPE_H261: if (SvOpenCodec (SV_H261_DECODE, &Info->Svh)!=SvErrorNone) { SlibClose((SlibHandle_t)Info); return(SlibErrorUnsupportedFormat); } Info->VideoCodecState=SLIB_CODEC_STATE_OPEN; _SlibDebug(_DEBUG_,printf("VideoCodecState=OPEN\n")); break; #endif /* H261_SUPPORT */ #ifdef H263_SUPPORT case SLIB_TYPE_H263: if (SvOpenCodec (SV_H263_DECODE, &Info->Svh)!=SvErrorNone) { SlibClose((SlibHandle_t)Info); return(SlibErrorUnsupportedFormat); } Info->VideoCodecState=SLIB_CODEC_STATE_OPEN; _SlibDebug(_DEBUG_,printf("VideoCodecState=OPEN\n")); break; #endif /* H263_SUPPORT */ #ifdef HUFF_SUPPORT case SLIB_TYPE_SHUFF: if (SvOpenCodec (SV_HUFF_DECODE, &Info->Svh)!=SvErrorNone) { SlibClose((SlibHandle_t)Info); return(SlibErrorUnsupportedFormat); } Info->VideoCodecState=SLIB_CODEC_STATE_OPEN; _SlibDebug(_DEBUG_,printf("VideoCodecState=OPEN\n")); break; #endif /* HUFF_SUPPORT */ #ifdef JPEG_SUPPORT case SLIB_TYPE_JPEG: case SLIB_TYPE_MJPG: if (SvOpenCodec (SV_JPEG_DECODE, &Info->Svh)!=SvErrorNone) { SlibClose((SlibHandle_t)Info); return(SlibErrorUnsupportedFormat); } Info->VideoCodecState=SLIB_CODEC_STATE_OPEN; _SlibDebug(_DEBUG_,printf("VideoCodecState=OPEN\n")); break; #endif /* JPEG_SUPPORT */ } /* VideoType */ } else return(SlibErrorBadMode); *stype = Info->Type; return(SlibErrorNone); } SlibStatus_t SlibAddBuffer(SlibHandle_t handle, SlibDataType_t dtype, void *buffer, unsigned dword bufsize) { SvStatus_t status; SlibInfo_t *Info=(SlibInfo_t *)handle; SlibPin_t *dstpin; if (!handle) return(SlibErrorBadHandle); dstpin = slibGetPin(Info, dtype); if (dstpin==NULL || buffer==NULL) return(SlibErrorBadArgument); if (Info->SlibCB) { status=slibManageUserBuffer(Info, buffer, bufsize, NULL); if (status!=SlibErrorNone) return(status); status=slibAddBufferToPin(dstpin, buffer, bufsize, SLIB_TIME_NONE); } else if (!SlibValidBuffer(buffer)) { /* we need to create a SLIB allocated buffer to copy the * output to and then add to the compressed data pin */ unsigned char *bufptr=SlibAllocBuffer(bufsize); if (!bufptr) return(SlibErrorMemory); memcpy(bufptr, buffer, bufsize); status=slibAddBufferToPin(dstpin, bufptr, bufsize, SLIB_TIME_NONE); } else status=slibAddBufferToPin(dstpin, buffer, bufsize, SLIB_TIME_NONE); if (Info->Mode==SLIB_MODE_DECOMPRESS) { ScBitstream_t *BS; Info->IOError=FALSE; /* reset end-of-input flags in bitstream objects */ if (Info->Svh) { BS=SvGetDataSource(Info->Svh); if (BS && BS->EOI) ScBSReset(BS); } if (Info->Sah) { BS=SaGetDataSource(Info->Sah); if (BS && BS->EOI) ScBSReset(BS); } } return(status); } SlibStatus_t SlibAddBufferEx(SlibHandle_t handle, SlibDataType_t dtype, void *buffer, unsigned dword bufsize, void *userdata) { SvStatus_t status; SlibInfo_t *Info=(SlibInfo_t *)handle; SlibPin_t *dstpin; if (!handle) return(SlibErrorBadHandle); dstpin = slibGetPin(Info, dtype); if (dstpin==NULL || buffer==NULL) return(SlibErrorBadArgument); status=slibManageUserBuffer(Info, buffer, bufsize, userdata); if (status!=SlibErrorNone) return(status); status=slibAddBufferToPin(dstpin, buffer, bufsize, SLIB_TIME_NONE); return(status); } SlibStatus_t slibStartVideo(SlibInfo_t *Info, SlibBoolean_t fillbuf) { SvStatus_t status=SvErrorNone; _SlibDebug(_VERBOSE_,printf("slibStartVideo()\n") ); if (Info->VideoCodecState==SLIB_CODEC_STATE_NONE || Info->VideoCodecState==SLIB_CODEC_STATE_BEGUN) { _SlibDebug(_DEBUG_,ScDebugPrintf(Info->dbg,"slibStartVideo(filebuf=%d) %s\n", fillbuf,Info->VideoCodecState==SLIB_CODEC_STATE_NONE ? "NONE" : "BEGUN") ); return(SlibErrorNone); } if (Info->VideoCodecState==SLIB_CODEC_STATE_OPEN) { _SlibDebug(_DEBUG_,ScDebugPrintf(Info->dbg,"slibStartVideo(filebuf=%d) OPEN\n", fillbuf)); if (Info->Mode==SLIB_MODE_DECOMPRESS) { if (Info->Type==SLIB_TYPE_YUV_AVI) { Info->VideoCodecState=SLIB_CODEC_STATE_BEGUN; _SlibDebug(_DEBUG_,ScDebugPrintf(Info->dbg,"VideoCodecState=BEGUN\n")); } else if (Info->Svh) { _SlibDebug(_DEBUG_,ScDebugPrintf(Info->dbg,"SvRegisterCallback()\n") ); status = SvRegisterCallback(Info->Svh, VDecompressCallback, (void *)Info); /* if codec is not bitstreaming, don't use callbacks */ if (status==SlibErrorNone && SvGetDataSource(Info->Svh)==NULL) { _SlibDebug(_DEBUG_,ScDebugPrintf(Info->dbg,"SvRegisterCallback(NULL)\n") ); status = SvRegisterCallback(Info->Svh, NULL, NULL); } _SlibDebug(_WARN_ && status!=SvErrorNone, ScDebugPrintf(Info->dbg,"SvRegisterCallback() %s\n", ScGetErrorStr(status)) ); Info->VideoCodecState=SLIB_CODEC_STATE_INITED; _SlibDebug(_DEBUG_,printf("VideoCodecState=INITED\n")); } } else if (Info->Mode==SLIB_MODE_COMPRESS) { if (Info->TotalBitRate==0) { #ifdef MPEG_SUPPORT if (Info->Type==SLIB_TYPE_MPEG_SYSTEMS || /* default to 1XCDROM rate */ Info->Type==SLIB_TYPE_MPEG_SYSTEMS_MPEG2) SlibSetParamInt((SlibHandle_t)Info, SLIB_STREAM_ALL, SLIB_PARAM_BITRATE, 44100*16*2); #endif slibValidateBitrates(Info); /* update bitrates */ } if (Info->Svh) { status = SvRegisterCallback(Info->Svh, VCompressCallback, (void *)Info); _SlibDebug(_WARN_ && status!=SvErrorNone, ScDebugPrintf(Info->dbg,"SvRegisterCallback() %s\n", ScGetErrorStr(status)) ); /* if codec is not bitstreaming, don't use callbacks */ if (status==SlibErrorNone && SvGetDataDestination(Info->Svh)==NULL) status = SvRegisterCallback(Info->Svh, NULL, NULL); Info->VideoCodecState=SLIB_CODEC_STATE_INITED; _SlibDebug(_DEBUG_,ScDebugPrintf(Info->dbg,"VideoCodecState=BEGUN\n")); } } } if (Info->VideoCodecState==SLIB_CODEC_STATE_INITED || Info->VideoCodecState==SLIB_CODEC_STATE_REPOSITIONING) { _SlibDebug(_DEBUG_,ScDebugPrintf(Info->dbg, "slibStartVideo(fillbuf=%d) INITED || REPOSITIONING\n",fillbuf)); if (Info->Mode==SLIB_MODE_DECOMPRESS) { if (Info->Type==SLIB_TYPE_YUV_AVI) { if (Info->CompVideoFormat->biCompression != Info->VideoFormat->biCompression && Info->Multibuf==NULL) { Info->MultibufSize=Info->ImageSize; Info->Multibuf = SlibAllocSharedBuffer(Info->MultibufSize, NULL); } } else if (Info->Svh) { int mbufsize; if (1) /* fillbuf && Info->CodecVideoFormat) */ { Info->CodecVideoFormat->biCompression= SlibGetParamInt((SlibHandle_t)Info, SLIB_STREAM_MAINVIDEO, SLIB_PARAM_NATIVEVIDEOFORMAT); if (Info->CodecVideoFormat->biCompression==0) Info->CodecVideoFormat->biCompression= Info->VideoFormat->biCompression; } else { Info->CodecVideoFormat->biCompression= Info->VideoFormat->biCompression; Info->CodecVideoFormat->biBitCount= Info->VideoFormat->biBitCount; } slibValidateVideoParams(Info); _SlibDebug(_DEBUG_, ScDebugPrintf(Info->dbg, "SvDecompressBegin(%c%c%c%c/%d bits,%c%c%c%c/%d bits)\n", (Info->CompVideoFormat->biCompression)&0xFF, (Info->CompVideoFormat->biCompression>>8)&0xFF, (Info->CompVideoFormat->biCompression>>16)&0xFF, (Info->CompVideoFormat->biCompression>>24)&0xFF, Info->CompVideoFormat->biBitCount, (Info->CodecVideoFormat->biCompression)&0xFF, (Info->CodecVideoFormat->biCompression>>8)&0xFF, (Info->CodecVideoFormat->biCompression>>16)&0xFF, (Info->CodecVideoFormat->biCompression>>24)&0xFF, Info->CodecVideoFormat->biBitCount) ); status=SvDecompressBegin(Info->Svh, Info->CompVideoFormat, Info->CodecVideoFormat); if (status==SvErrorNone) { Info->KeySpacing=(int)SvGetParamInt(Info->Svh, SV_PARAM_KEYSPACING); Info->SubKeySpacing=(int)SvGetParamInt(Info->Svh, SV_PARAM_SUBKEYSPACING); Info->VideoCodecState=SLIB_CODEC_STATE_BEGUN; Info->HeaderProcessed=TRUE; /* we must have processed header info */ _SlibDebug(_DEBUG_,ScDebugPrintf(Info->dbg,"VideoCodecState=BEGUN\n")); } else if (status==SvErrorEndBitstream) return(SlibErrorNoBeginning); else { _SlibDebug(_WARN_, ScDebugPrintf(Info->dbg,"SvDecompressBegin() %s\n", ScGetErrorStr(status)) ); return(SlibErrorUnsupportedFormat); } _SlibDebug(_DEBUG_, ScDebugPrintf(Info->dbg,"SvGetDecompressSize\n") ); SvGetDecompressSize(Info->Svh, &mbufsize); if (Info->Multibuf==NULL || Info->MultibufSizeMultibuf) SlibFreeBuffer(Info->Multibuf); Info->MultibufSize=mbufsize; Info->Multibuf = SlibAllocSharedBuffer(Info->MultibufSize, NULL); } } } else if (Info->Mode==SLIB_MODE_COMPRESS && Info->Svh) { status=SvCompressBegin(Info->Svh, Info->CodecVideoFormat, Info->CompVideoFormat); if (status==SvErrorNone) { Info->KeySpacing=(int)SvGetParamInt(Info->Svh, SV_PARAM_KEYSPACING); Info->SubKeySpacing=(int)SvGetParamInt(Info->Svh, SV_PARAM_SUBKEYSPACING); Info->VideoCodecState=SLIB_CODEC_STATE_BEGUN; _SlibDebug(_DEBUG_,ScDebugPrintf(Info->dbg,"VideoCodecState=BEGUN\n")); } else { _SlibDebug(_WARN_, ScDebugPrintf(Info->dbg,"SvCompressBegin() %s\n", ScGetErrorStr(status)) ); return(SlibErrorUnsupportedFormat); } } } if (Info->VideoCodecState==SLIB_CODEC_STATE_BEGUN) return(SlibErrorNone); else return(SlibErrorInit); } static SlibStatus_t slibStartAudio(SlibInfo_t *Info) { SvStatus_t status=SvErrorNone; _SlibDebug(_VERBOSE_,printf("slibStartAudio()\n") ); if (Info->AudioCodecState==SLIB_CODEC_STATE_NONE || Info->AudioCodecState==SLIB_CODEC_STATE_BEGUN) { _SlibDebug(_DEBUG_,printf("slibStartAudio() %s\n", Info->AudioCodecState==SLIB_CODEC_STATE_NONE ? "NONE" : "BEGUN") ); return(SlibErrorNone); } if (Info->AudioCodecState==SLIB_CODEC_STATE_OPEN) { _SlibDebug(_DEBUG_,printf("slibStartAudio() OPEN\n")); if (Info->Sah) { if (Info->Mode==SLIB_MODE_DECOMPRESS) { status = SaRegisterCallback(Info->Sah, ADecompressCallback, (void *)Info); if (status!=SaErrorNone) { _SlibDebug(_WARN_, printf("SaRegisterCallback() ", ScGetErrorStr(status)) ); return(SlibErrorInternal); } status = SaSetDataSource(Info->Sah, SA_USE_BUFFER_QUEUE, 0, (void *)Info, 0); _SlibDebug(_WARN_ && status!=SaErrorNone, printf("SaSetDataSource() ", ScGetErrorStr(status)) ); Info->AudioCodecState=SLIB_CODEC_STATE_INITED; _SlibDebug(_DEBUG_,printf("AudioCodecState=INITED\n")); } else if (Info->Mode==SLIB_MODE_COMPRESS) { if (Info->TotalBitRate==0) { #ifdef MPEG_SUPPORT /* default to 1X CDROM rate */ if (Info->Type==SLIB_TYPE_MPEG_SYSTEMS || Info->Type==SLIB_TYPE_MPEG_SYSTEMS_MPEG2) SlibSetParamInt((SlibHandle_t)Info, SLIB_STREAM_ALL, SLIB_PARAM_BITRATE, 44100*16*2); #endif slibValidateBitrates(Info); /* update bitrates */ } status = SaRegisterCallback(Info->Sah, ACompressCallback, (void *)Info); _SlibDebug(_WARN_ && status!=SaErrorNone, printf("SaRegisterCallback() %s\n", ScGetErrorStr(status)) ); status = SaSetDataDestination(Info->Sah, SA_USE_BUFFER_QUEUE, 0, (void *)Info, 0); _SlibDebug(_WARN_ && status!=SaErrorNone, printf("SaSetDataDestination() %s\n", ScGetErrorStr(status)) ); Info->AudioCodecState=SLIB_CODEC_STATE_INITED; _SlibDebug(_DEBUG_,printf("AudioCodecState=INITED\n")); } } } if (Info->AudioCodecState==SLIB_CODEC_STATE_INITED || Info->AudioCodecState==SLIB_CODEC_STATE_REPOSITIONING) { _SlibDebug(_DEBUG_,printf("slibStartAudio() INITED || REPOSITIONING\n")); if (Info->Sah) { if (Info->Mode==SLIB_MODE_DECOMPRESS) { Info->AudiobufUsed=0; /* don't want codec to search through to much data for start */ status=SaDecompressBegin(Info->Sah, Info->CompAudioFormat, Info->AudioFormat); if (status==SaErrorNone) { Info->AudioCodecState=SLIB_CODEC_STATE_BEGUN; _SlibDebug(_DEBUG_,printf("AudioCodecState=BEGUN\n")); } else if (status==SlibErrorNoBeginning) return(SlibErrorEndOfStream); else { _SlibDebug(_WARN_, printf("SaDecompressBegin() %s\n", ScGetErrorStr(status)) ); return(SlibErrorUnsupportedFormat); } } else if (Info->Mode==SLIB_MODE_COMPRESS) { status=SaCompressBegin(Info->Sah, Info->AudioFormat, Info->CompAudioFormat); if (status==SvErrorNone) { Info->AudioCodecState=SLIB_CODEC_STATE_BEGUN; _SlibDebug(_DEBUG_,printf("AudioCodecState=BEGUN\n")); } else { _SlibDebug(_WARN_, printf("SaCompressBegin() %s\n", ScGetErrorStr(status)) ); return(SlibErrorUnsupportedFormat); } } } } return(SlibErrorNone); } SlibStatus_t SlibRegisterVideoBuffer(SlibHandle_t handle, void *buffer, unsigned dword bufsize) { SvStatus_t status; SlibInfo_t *Info=(SlibInfo_t *)handle; dword mbufsize; if (!handle) return(SlibErrorBadHandle); if (Info->Multibuf) SlibFreeBuffer(Info->Multibuf); Info->MultibufSize=bufsize; Info->Multibuf = buffer; status=slibManageUserBuffer(Info, buffer, bufsize, NULL); if (Info->Svh) { SvGetDecompressSize(Info->Svh, &mbufsize); if (bufsize<(unsigned dword)mbufsize) return(SlibErrorBufSize); } return(status); } SlibStatus_t SlibReadData(SlibHandle_t handle, SlibStream_t stream, void **databuf, unsigned dword *databufsize, SlibStream_t *readstream) { SlibInfo_t *Info=(SlibInfo_t *)handle; int pinid; SlibPin_t *pin; SlibTime_t ptimestamp; _SlibDebug(_VERBOSE_, printf("SlibReadDATA()\n") ); if (!handle) return(SlibErrorBadHandle); if (!databuf) /* we're querying to find out how much data is queued */ { if (!databufsize) return(SlibErrorBadArgument); if (Info->Mode==SLIB_MODE_COMPRESS) pinid=SLIB_DATA_COMPRESSED; else if (stream==SLIB_STREAM_MAINVIDEO) pinid=SLIB_DATA_VIDEO; else if (stream==SLIB_STREAM_MAINAUDIO) pinid=SLIB_DATA_AUDIO; else { *databufsize=(unsigned dword)slibDataOnPins(Info); /* get amount of data on all pins */ return(SlibErrorNone); } *databufsize=(unsigned dword)slibDataOnPin(Info, SLIB_DATA_COMPRESSED); return(SlibErrorNone); } if (Info->Mode==SLIB_MODE_COMPRESS) { pinid=SLIB_DATA_COMPRESSED; stream=SLIB_STREAM_ALL; /* flush out all compressed data */ if (Info->Sah) ScBSFlush(SaGetDataDestination(Info->Sah)); if (Info->Svh) ScBSFlush(SvGetDataDestination(Info->Svh)); } else /* SLIB_MODE_DECOMPRESS */ { if (stream==SLIB_STREAM_ALL && (Info->AudioStreams || Info->VideoStreams)) { if (Info->AudioStreams==0) /* there's only video */ stream=SLIB_STREAM_MAINVIDEO; else if (Info->VideoStreams==0) /* there's only audio */ stream=SLIB_STREAM_MAINAUDIO; else if (slibDataOnPin(Info, SLIB_DATA_AUDIO)> slibDataOnPin(Info, SLIB_DATA_VIDEO)) /* more audio than video */ stream=SLIB_STREAM_MAINAUDIO; else stream=SLIB_STREAM_MAINVIDEO; } switch (stream) /* translate stream to pin */ { case SLIB_STREAM_MAINVIDEO: pinid=SLIB_DATA_VIDEO; break; case SLIB_STREAM_MAINAUDIO: pinid=SLIB_DATA_AUDIO; break; default: return(SlibErrorBadStream); } } if (readstream) *readstream=stream; pin=slibLoadPin(Info, pinid); if (pin==NULL) return(Info->Mode==SLIB_MODE_COMPRESS?SlibErrorNoData:SlibErrorBadStream); if (stream==SLIB_STREAM_MAINVIDEO && Info->Mode==SLIB_MODE_DECOMPRESS && Info->VideoPTimeCode==SLIB_TIME_NONE && SlibTypeIsMPEG(Info->Type)) { /* search from GOP start */ dword i, iend; SlibTime_t nexttime; unsigned char *tmpbuf, *prevbuf=NULL; unsigned dword tmpsize, bytessearched=0; tmpbuf=SlibGetBuffer(Info, pinid, &tmpsize, &ptimestamp); if (tmpbuf==NULL) return(SlibErrorEndOfStream); do { for (i=0, iend=tmpsize-3; iMode==SLIB_MODE_DECOMPRESS) { if (ptimestamp!=SLIB_TIME_NONE) switch (stream) /* set presentation timecodes */ { case SLIB_STREAM_MAINVIDEO: Info->VideoPTimeCode=ptimestamp; _SlibDebug(_TIMECODE_ || _VERBOSE_, printf("SlibReadData() VideoPTimeCode=%ld\n", ptimestamp) ); break; case SLIB_STREAM_MAINAUDIO: Info->AudioPTimeCode=ptimestamp; _SlibDebug(_TIMECODE_ || _VERBOSE_, printf("SlibReadData() AudioPTimeCode=%ld\n", ptimestamp) ); break; } else if (stream==SLIB_STREAM_MAINVIDEO && Info->VideoPTimeCode==SLIB_TIME_NONE) Info->VideoPTimeCode=SLIB_TIME_UNKNOWN; } if (*databuf==NULL || *databufsize==0) { if (!slibDataOnPin(Info, pinid)) return(SlibErrorEndOfStream); else return(SlibErrorReading); } return(SlibErrorNone); } SlibStatus_t SlibReadVideo(SlibHandle_t handle, SlibStream_t stream, void **videobuf, unsigned dword *videobufsize) { SlibInfo_t *Info=(SlibInfo_t *)handle; SvStatus_t status=SvErrorNone; unsigned char *imagebuf=NULL; SlibTime_t startvideotime; _SlibDebug(_VERBOSE_, printf("SlibReadVideo()\n") ); if (!handle) return(SlibErrorBadHandle); if (!videobuf) return(SlibErrorBadArgument); /* imagesize=(Info->Width*Info->Height*3)/2; if (videobufsizeMode!=SLIB_MODE_DECOMPRESS) return(SlibErrorBadMode); if (Info->VideoFormat==NULL || Info->CodecVideoFormat==NULL || Info->CompVideoFormat==NULL) return(SlibErrorUnsupportedFormat); if ((status=slibStartVideo(Info, (SlibBoolean_t)((*videobuf==NULL)?FALSE:TRUE))) !=SlibErrorNone) return(status); startvideotime=Info->VideoTimeStamp; switch(Info->VideoType) { #ifdef MPEG_SUPPORT case SLIB_TYPE_MPEG1_VIDEO: case SLIB_TYPE_MPEG2_VIDEO: do { _SlibDebug(_DEBUG_, printf("SvDecompressMPEG()\n") ); status = SvDecompressMPEG(Info->Svh, Info->Multibuf, Info->MultibufSize, &imagebuf); _SlibDebug(_WARN_ && status!=SvErrorNone, printf("SvDecompressMPEG() %s\n", ScGetErrorStr(status)) ); } while(status == SvErrorNotDecompressable); if (status==SvErrorNone) SlibAllocSubBuffer(imagebuf, Info->CodecVideoFormat->biSizeImage); _SlibDebug(_SEEK_>1, printf("timecode=%d ms framenum=%d\n", SvGetParamInt(Info->Svh, SV_PARAM_CALCTIMECODE), SvGetParamInt(Info->Svh, SV_PARAM_FRAME) ) ); break; #endif /* MPEG_SUPPORT */ #ifdef H261_SUPPORT case SLIB_TYPE_H261: do { _SlibDebug(_DEBUG_, ScDebugPrintf(Info->dbg,"SvDecompressH261()\n") ); status = SvDecompressH261(Info->Svh, Info->Multibuf, Info->MultibufSize, &imagebuf); } while(status == SvErrorNotDecompressable); if (status==SvErrorNone) SlibAllocSubBuffer(imagebuf, Info->CodecVideoFormat->biSizeImage); break; #endif /* H261_SUPPORT */ #ifdef H263_SUPPORT case SLIB_TYPE_H263: _SlibDebug(_DEBUG_, ScDebugPrintf(Info->dbg,"SvDecompress(%d bytes)\n", Info->MultibufSize) ); status=SvDecompress(Info->Svh, NULL, 0, Info->Multibuf, Info->MultibufSize); imagebuf=Info->Multibuf; if (status==SvErrorNone) SlibAllocSubBuffer(imagebuf, Info->CodecVideoFormat->biSizeImage); break; #endif /* H263_SUPPORT */ #ifdef JPEG_SUPPORT case SLIB_TYPE_JPEG: case SLIB_TYPE_MJPG: { unsigned dword bufsize; unsigned char *buf; buf=SlibGetBuffer(Info, SLIB_DATA_VIDEO, &bufsize, NULL); if (buf) { /* ScDumpChar(buf, 10000, 0); */ _SlibDebug(_DEBUG_, ScDebugPrintf(Info->dbg,"SvDecompress(%d bytes)\n", bufsize) ); status=SvDecompress(Info->Svh, buf, bufsize, Info->Multibuf, Info->MultibufSize); imagebuf=Info->Multibuf; SlibFreeBuffer(buf); } else status=SvErrorForeign; if (status==SvErrorNone) SlibAllocSubBuffer(imagebuf, Info->CodecVideoFormat->biSizeImage); } break; #endif /* JPEG_SUPPORT */ #ifdef HUFF_SUPPORT case SLIB_TYPE_SHUFF: if (*videobuf==NULL) { if (Info->Imagebuf==NULL && (Info->Imagebuf=SlibAllocBuffer(Info->ImageSize))==NULL) return(SlibErrorMemory); imagebuf=Info->Imagebuf; } else imagebuf=*videobuf; do { _SlibDebug(_DEBUG_, ScDebugPrintf(Info->dbg,"SvDecompress()\n") ); status=SvDecompress(Info->Svh, NULL, 0, imagebuf, Info->CodecVideoFormat->biSizeImage); } while(status == SvErrorNotDecompressable); if (status==SvErrorNone) SlibAllocSubBuffer(imagebuf, Info->CodecVideoFormat->biSizeImage); break; #endif /* HUFF_SUPPORT */ case SLIB_TYPE_RASTER: case SLIB_TYPE_YUV: if (*videobuf && videobufsize && *videobufsize==0) return(SlibErrorBadArgument); imagebuf=SlibGetBuffer(Info, SLIB_DATA_VIDEO, videobufsize, NULL); if (*videobufsize==0) status=SvErrorEndBitstream; _SlibDebug(_DEBUG_, ScDebugPrintf(Info->dbg,"Video frame size = %d ImageSize=%d\n", *videobufsize, Info->ImageSize) ); break; default: return(SlibErrorUnsupportedFormat); } if (status==SvErrorNone) { /* format conversion */ if (Info->Sch==NULL) /* start the format converter */ { if (Info->Svh) /* compressed video format */ { unsigned dword fourcc=(unsigned dword)SvGetParamInt(Info->Svh, SV_PARAM_FINALFORMAT); if (fourcc) { Info->CodecVideoFormat->biCompression=fourcc; Info->CodecVideoFormat->biBitCount= (WORD)slibCalcBits(fourcc, Info->CodecVideoFormat->biBitCount); } } else /* uncompressed video format */ memcpy(Info->CodecVideoFormat, Info->CompVideoFormat, sizeof(BITMAPINFOHEADER)); if (SconOpen(&Info->Sch, SCON_MODE_VIDEO, (void *)Info->CodecVideoFormat, (void *)Info->VideoFormat) !=SconErrorNone) return(SlibErrorUnsupportedFormat); if (Info->Stride) SconSetParamInt(Info->Sch, SCON_OUTPUT, SCON_PARAM_STRIDE, Info->Stride); } if (SconIsSame(Info->Sch) && *videobuf == NULL) /* no conversion */ *videobuf=imagebuf; else { if (*videobuf == NULL && (*videobuf=SlibAllocBuffer(Info->ImageSize))==NULL) return(SlibErrorMemory); if (SconConvert(Info->Sch, imagebuf, Info->CodecVideoFormat->biSizeImage, *videobuf, Info->ImageSize) != SconErrorNone) { SlibFreeBuffer(imagebuf); /* free decompressed image */ return(SlibErrorUnsupportedFormat); } SlibFreeBuffer(imagebuf); /* free decompressed image */ } *videobufsize = Info->ImageSize; /* update stats */ if (Info->stats && Info->stats->Record) Info->stats->FramesProcessed++; if (startvideotime==Info->VideoTimeStamp) /* video time hasn't changed */ slibAdvancePositions(Info, 1); } else { if (status==ScErrorEndBitstream || !slibDataOnPin(Info, SLIB_DATA_VIDEO)) { if (Info->FileSize>0 && !Info->VideoLengthKnown) slibUpdateLengths(Info); _SlibDebug(_WARN_, ScDebugPrintf(Info->dbg,"SlibReadVideo() %s\n", ScGetErrorStr(status)) ); return(SlibErrorEndOfStream); } _SlibDebug(_WARN_, printf("SlibReadVideo() %s\n", ScGetErrorStr(status)) ); return(SlibErrorReading); } return(SlibErrorNone); } SlibStatus_t SlibReadAudio(SlibHandle_t handle, SlibStream_t stream, void *audiobuf, unsigned dword *audiobufsize) { SlibInfo_t *Info=(SlibInfo_t *)handle; SvStatus_t status=SaErrorNone; unsigned dword totalbytes=0, bytes_since_timeupdate=0; SlibTime_t startaudiotime; #ifdef _SLIBDEBUG_ SlibTime_t calcaudiotime; #endif _SlibDebug(_VERBOSE_, printf("SlibReadAudio(audiobufsize=%d, time=%d)\n", *audiobufsize, Info->AudioTimeStamp) ); if (!handle) return(SlibErrorBadHandle); if (Info->Mode!=SLIB_MODE_DECOMPRESS) return(SlibErrorBadMode); if (Info->AudioFormat==NULL) return(SlibErrorUnsupportedFormat); if ((status=slibStartAudio(Info))!=SlibErrorNone) return(status); #ifdef _SLIBDEBUG_ calcaudiotime=Info->AudioTimeStamp; #endif startaudiotime=Info->AudioTimeStamp; switch(Info->AudioType) { case SLIB_TYPE_PCM: totalbytes=slibFillBufferFromPin(Info, slibGetPin(Info, SLIB_DATA_AUDIO), (unsigned char *)audiobuf, *audiobufsize, NULL); if (totalbytes==0) status=ScErrorEndBitstream; *audiobufsize = totalbytes; bytes_since_timeupdate = totalbytes; break; #ifdef MPEG_SUPPORT case SLIB_TYPE_MPEG1_AUDIO: { unsigned dword bytes; /* see if some bytes of audio are left in the temp audio buffer */ if (Info->Audiobuf && Info->AudiobufUsed>0) { _SlibDebug(_DEBUG_, printf("SlibReadAudio() Audiobuf contains %d bytes\n", Info->AudiobufUsed) ); if (*audiobufsize>=Info->AudiobufUsed) { memcpy(audiobuf, Info->Audiobuf, Info->AudiobufUsed); totalbytes=Info->AudiobufUsed; Info->AudiobufUsed=0; } else { memcpy(audiobuf, Info->Audiobuf, *audiobufsize); totalbytes=*audiobufsize; Info->AudiobufUsed-=*audiobufsize; memcpy(Info->Audiobuf, Info->Audiobuf+*audiobufsize, Info->AudiobufUsed); } } /* need to alloc a temp audio buffer? */ if (!Info->Audiobuf || Info->AudiobufSize< *audiobufsize+MPEG1_AUDIO_FRAME_SIZE*4) { unsigned char *newbuf; /* enlarge Audiobuf or alloc it for the first time */ _SlibDebug(_DEBUG_, printf("SlibReadAudio() enlarging Audiobuf: %d->%d bytes\n", Info->AudiobufSize,*audiobufsize+MPEG1_AUDIO_FRAME_SIZE*4) ); newbuf=SlibAllocBuffer(*audiobufsize+MPEG1_AUDIO_FRAME_SIZE*4); if (!newbuf) return(SlibErrorMemory); Info->AudiobufSize=*audiobufsize+MPEG1_AUDIO_FRAME_SIZE*4; if (Info->Audiobuf) SlibFreeBuffer(Info->Audiobuf); Info->Audiobuf=newbuf; Info->AudiobufUsed=0; } if (*audiobufsize>=MPEG1_AUDIO_FRAME_SIZE*4) { unsigned dword stopbytes=*audiobufsize-(MPEG1_AUDIO_FRAME_SIZE*4)+1; while (status==SaErrorNone && totalbytesSah, NULL, 0, (unsigned char *)audiobuf+totalbytes, &bytes); _SlibDebug(_DEBUG_, printf("SaDecompress() out: bytes=%d\n", bytes) ); totalbytes += bytes; if (Info->AudioTimeStamp!=startaudiotime) { startaudiotime=Info->AudioTimeStamp; bytes_since_timeupdate=bytes; } else bytes_since_timeupdate+=bytes; } } if (totalbytes<*audiobufsize && status==SaErrorNone) { unsigned dword neededbytes=*audiobufsize-totalbytes; while (status==SaErrorNone && Info->AudiobufUsedSah, NULL, 0, (unsigned char *)Info->Audiobuf+Info->AudiobufUsed, &bytes); _SlibDebug(_DEBUG_, printf("SaDecompress() out, %d bytes\n", bytes) ); Info->AudiobufUsed += bytes; } if (Info->AudiobufUsed>0) { if (Info->AudiobufUsed>neededbytes) /* complete buffer returned */ { memcpy((unsigned char*)audiobuf+totalbytes, Info->Audiobuf, neededbytes); Info->AudiobufUsed-=neededbytes; memcpy(Info->Audiobuf, Info->Audiobuf+neededbytes, Info->AudiobufUsed); totalbytes+=neededbytes; bytes_since_timeupdate+=neededbytes; } else /* partially filled buffer */ { memcpy((unsigned char*)audiobuf+totalbytes, Info->Audiobuf, Info->AudiobufUsed); totalbytes+=Info->AudiobufUsed; Info->AudiobufUsed=0; } } } *audiobufsize = totalbytes; _SlibDebug(_WARN_>1 && totalbytes>0 && totalbytes!=*audiobufsize, printf("SlibReadAudio(audiobufsize=%d bytes) totalbytes=%d\n", *audiobufsize, totalbytes) ); } break; #endif /* MPEG_SUPPORT */ #ifdef AC3_SUPPORT case SLIB_TYPE_AC3_AUDIO: { unsigned dword bytes; unsigned int framesize; unsigned int buffersize; int samplesize; int buffers; unsigned char *pointers[3]; int i; if (Info->Channels>2) { framesize = AC3_FRAME_SIZE*((Info->BitsPerSample+7)/8) *Info->Channels; samplesize=Info->Channels*((Info->BitsPerSample+7)/8); buffers = (Info->Channels+1)/2; buffersize = (*audiobufsize/samplesize/buffers)*samplesize; for(i=0; i=framesize) { while (status==SaErrorNone && totalbytesSah, NULL, 0, pointers, &bytes); _SlibDebug(_DEBUG_, printf("SaDecompress() out, %d bytes\n", bytes) ); for(i=0;iAudioTimeStamp!=startaudiotime) { startaudiotime=Info->AudioTimeStamp; bytes_since_timeupdate=bytes; } else bytes_since_timeupdate+=bytes; } } } else { while (status==SaErrorNone && totalbytes<*audiobufsize) { bytes = *audiobufsize - totalbytes; _SlibDebug(_DEBUG_, printf("SaDecompress() in totalbytes=%d\n", totalbytes) ); status = SaDecompress(Info->Sah, NULL, 0, (unsigned char *)audiobuf+totalbytes, &bytes); _SlibDebug(_DEBUG_, printf("SaDecompress() out, %d bytes\n", bytes) ); totalbytes += bytes; if (Info->AudioTimeStamp!=startaudiotime) { startaudiotime=Info->AudioTimeStamp; bytes_since_timeupdate=bytes; } else bytes_since_timeupdate+=bytes; } } /* ** NOTE: The semantics are different here ** we return the size of just one stereo pair's buffer */ *audiobufsize = totalbytes; _SlibDebug(_WARN_>1 && totalbytes>0 && totalbytes!=*audiobufsize, printf("SlibReadAudio(audiobufsize=%d bytes) totalbytes=%d\n", *audiobufsize, totalbytes) ); } break; #endif /* AC3_SUPPORT */ #ifdef G723_SUPPORT case SLIB_TYPE_G723: //G723 decompresses in multiples of 480 samples. //To eliminate cumbersome buffer calculations, // Always fill the output buffer up to multiples // of 480 samples.To do this we iterate basically // the below "while" loop "audiobufsize/480 times. { int iTimes = (int)*audiobufsize/480; int iLoop =0; unsigned dword bytes; if (slibInSyncMode(Info)) { /* in synchronous mode we can't decompress past last frame * otherwise we may lose a frame */ int iMaxTimes=(int)(slibDataOnPin(Info, SLIB_DATA_COMPRESSED)+ slibDataOnPin(Info, SLIB_DATA_AUDIO))/ SlibGetParamInt(handle, stream, SLIB_PARAM_MININPUTSIZE); if (iTimes>iMaxTimes) iTimes=iMaxTimes; } while (status==SaErrorNone && iLoopSah, NULL, 0, (unsigned char *)audiobuf+totalbytes, &bytes); _SlibDebug(_DEBUG_, printf("SaDecompress() out, %d bytes\n", bytes) ); totalbytes += bytes; iLoop++; if (Info->AudioTimeStamp!=startaudiotime) { startaudiotime=Info->AudioTimeStamp; bytes_since_timeupdate=bytes; } else bytes_since_timeupdate+=bytes; } *audiobufsize = totalbytes; _SlibDebug(_WARN_>1 && totalbytes>0 && totalbytes!=*audiobufsize, printf("SlibReadAudio(audiobufsize=%d bytes) totalbytes=%d\n", *audiobufsize, totalbytes) ); } break; #endif /* G723_SUPPORT */ default: *audiobufsize = 0; return(SlibErrorUnsupportedFormat); } /* as we're decompressing audiotime may be updated with timecodes */ if (Info->AudioTimeStamp==startaudiotime) Info->AudioTimeStamp = startaudiotime + (bytes_since_timeupdate*8000)/ (Info->SamplesPerSec*Info->BitsPerSample*Info->Channels); _SlibDebug(_TIMECODE_||_VERBOSE_, calcaudiotime += (*audiobufsize*8000)/ (Info->SamplesPerSec*Info->BitsPerSample*Info->Channels); printf("AudioTimeStamp=%ld calcaudiotime=%ld (diff=%ld)\n", Info->AudioTimeStamp, calcaudiotime, calcaudiotime-Info->AudioTimeStamp); Info->AudioTimeStamp = calcaudiotime ); _SlibDebug(_VERBOSE_||_TIMECODE_, printf("ReadAudio(%d) Time=%ld SamplesPerSec=%d BitsPerSample=%d Channels=%d\n", totalbytes, Info->AudioTimeStamp, Info->SamplesPerSec, Info->BitsPerSample, Info->Channels) ); /* Info->SystemTimeStamp=Info->AudioTimeStamp; */ if (status==SaErrorNone) return(SlibErrorNone); else if (status==ScErrorEndBitstream || status==ScErrorEOI) { if (*audiobufsize!=0) return(SlibErrorNone); else return(SlibErrorEndOfStream); } else { _SlibDebug(_WARN_ && status!=ScErrorEndBitstream && status!=ScErrorEOI, printf("SlibReadAudio() %s\n", ScGetErrorStr(status)) ); if (SlibIsEnd(handle, stream)) return(SlibErrorEndOfStream); return(SlibErrorReading); } } SlibStatus_t SlibWriteVideo(SlibHandle_t handle, SlibStream_t stream, void *videobuf, unsigned dword videobufsize) { int compsize=0; SlibInfo_t *Info=(SlibInfo_t *)handle; SvStatus_t status; _SlibDebug(_DEBUG_, printf("SlibWriteVideo()\n") ); if (!handle) return(SlibErrorBadHandle); if (!videobuf) return(SlibErrorBadArgument); if (videobufsize<(unsigned dword)Info->ImageSize) return(SlibErrorBufSize); if (Info->Mode!=SLIB_MODE_COMPRESS) return(SlibErrorBadMode); if (Info->VideoFormat==NULL || Info->CompVideoFormat==NULL) return(SlibErrorUnsupportedFormat); if (Info->IOError) return(SlibErrorWriting); if ((status=slibStartVideo(Info, FALSE))!=SlibErrorNone) return(status); if (Info->Sch==NULL) /* start the format converter */ { if (SconOpen(&Info->Sch, SCON_MODE_VIDEO, (void *)Info->VideoFormat, (void *)Info->CodecVideoFormat) !=SconErrorNone) return(SlibErrorUnsupportedFormat); } if (!SconIsSame(Info->Sch)) /* need a conversion */ { unsigned char *tmpbuf=NULL; if (Info->CodecImagebuf==NULL && (Info->CodecImagebuf=SlibAllocBuffer(Info->CodecImageSize))==NULL) return(SlibErrorMemory); if (SconConvert(Info->Sch, videobuf, Info->ImageSize, Info->CodecImagebuf, Info->CodecImageSize) != SconErrorNone) return(SlibErrorUnsupportedFormat); videobuf=Info->CodecImagebuf; videobufsize=Info->CodecImageSize; } switch(Info->Type) { #ifdef H261_SUPPORT case SLIB_TYPE_H261: status = SvCompress(Info->Svh, NULL, 0, videobuf, videobufsize, &compsize); break; #endif /* H261_SUPPORT */ #ifdef H263_SUPPORT case SLIB_TYPE_H263: status = SvCompress(Info->Svh, NULL, 0, videobuf, videobufsize, &compsize); break; #endif /* H263_SUPPORT */ #ifdef MPEG_SUPPORT case SLIB_TYPE_MPEG1_VIDEO: case SLIB_TYPE_MPEG2_VIDEO: case SLIB_TYPE_MPEG_SYSTEMS: case SLIB_TYPE_MPEG_SYSTEMS_MPEG2: status = SvCompress(Info->Svh, NULL, 0, videobuf, videobufsize, &compsize); break; #endif /* MPEG_SUPPORT */ #ifdef HUFF_SUPPORT case SLIB_TYPE_SHUFF: status = SvCompress(Info->Svh, NULL, 0, videobuf, videobufsize, &compsize); break; #endif /* HUFF_SUPPORT */ default: return(SlibErrorUnsupportedFormat); } if (status==SvErrorNone && !Info->IOError) { if (Info->stats && Info->stats->Record) Info->stats->FramesProcessed++; slibAdvancePositions(Info, 1); } else { _SlibDebug(_WARN_, printf("SlibWriteVideo() %s\n", ScGetErrorStr(status)) ); if (status==ScErrorEndBitstream || Info->IOError) return(SlibErrorEndOfStream); return(SlibErrorWriting); } return(SlibErrorNone); } SlibStatus_t SlibWriteAudio(SlibHandle_t handle, SlibStream_t stream, void *audiobuf, unsigned dword audiobufsize) { unsigned dword compsize=0; SlibInfo_t *Info=(SlibInfo_t *)handle; SvStatus_t status; _SlibDebug(_DEBUG_, printf("SlibAudioVideo()\n") ); if (!handle) return(SlibErrorBadHandle); if (!audiobuf) return(SlibErrorBadArgument); if (Info->Mode!=SLIB_MODE_COMPRESS) return(SlibErrorBadMode); if (Info->AudioFormat==NULL || Info->CompAudioFormat==NULL) { _SlibDebug(_VERBOSE_ || _WARN_, printf("SlibWriteAudio() Audio Formats not setup\n") ); return(SlibErrorUnsupportedFormat); } if (Info->IOError) return(SlibErrorWriting); if ((status=slibStartAudio(Info))!=SlibErrorNone) return(status); switch(Info->Type) { #ifdef MPEG_SUPPORT case SLIB_TYPE_MPEG1_AUDIO: case SLIB_TYPE_MPEG_SYSTEMS: case SLIB_TYPE_MPEG_SYSTEMS_MPEG2: { unsigned dword audiobytes; void *audiooutbuf=NULL; status=slibConvertAudio(Info, audiobuf, audiobufsize, Info->SamplesPerSec, Info->BitsPerSample, &audiooutbuf, &audiobytes, Info->AudioFormat->nSamplesPerSec, Info->AudioFormat->wBitsPerSample, Info->Channels); if (status!=SlibErrorNone) return(status); audiobuf=audiooutbuf; audiobufsize=audiobytes; if (Info->AudiobufUsed && Info->Audiobuf) /* left over audio data */ { if (Info->AudiobufSizeAudiobufUsed+audiobufsize) { unsigned char *newbuf; /* enlarge Audiobuf */ _SlibDebug(_DEBUG_, printf("enlarging Audiobuf: %d->%d bytes\n", Info->AudiobufSize,audiobufsize+4608) ); newbuf=SlibAllocBuffer(audiobufsize+4608); if (!newbuf) return(SlibErrorMemory); memcpy(newbuf, Info->Audiobuf, Info->AudiobufUsed); SlibFreeBuffer(Info->Audiobuf); Info->AudiobufSize+=audiobufsize; Info->Audiobuf=newbuf; } _SlibDebug(_DEBUG_, printf("Appending audio data: Info->AudiobufUsed=%d\n", Info->AudiobufUsed) ); memcpy(Info->Audiobuf+Info->AudiobufUsed, audiobuf, audiobufsize); audiobuf=Info->Audiobuf; audiobufsize+=Info->AudiobufUsed; audiobytes=audiobufsize; Info->AudiobufUsed=0; } status = SaCompress(Info->Sah, (unsigned char *)audiobuf, &audiobytes, NULL, &compsize); if (audiobytesAudiobuf) { Info->AudiobufSize=audiobufsize+(audiobufsize-audiobytes); Info->Audiobuf=SlibAllocBuffer(Info->AudiobufSize); if (!Info->Audiobuf) { Info->AudiobufSize=0; return(SlibErrorMemory); } } memcpy(Info->Audiobuf, (unsigned char *)audiobuf+audiobytes, audiobufsize-audiobytes); Info->AudiobufUsed=audiobufsize-audiobytes; } audiobufsize=audiobytes; /* actual amount written */ if (audiooutbuf) SlibFreeBuffer(audiooutbuf); } break; #endif /* MPEG_SUPPORT */ #ifdef G723_SUPPORT case SLIB_TYPE_G723: { unsigned int iNumBytesUnProcessed =0; unsigned int iNumBytesCompressed = 0; //You always compress in terms of frames (Frame is 480 bytes) //So,the files with sizes which are not exactly divisible by // 480 always leave some bytes at the end Unprocessed,Which is O.K //Check for any Unprocessed Audio stored in Temp buff //from the previous call to SlibWriteAudio. if (Info->AudiobufUsed && Info->Audiobuf) /* left over audio data */ { if (Info->AudiobufSize < Info->AudiobufUsed+audiobufsize) { unsigned char *newbuf; /* enlarge Audiobuf to new Size (Current size + left over audio)*/ _SlibDebug(_DEBUG_, printf("enlarging Audiobuf: %d->%d bytes\n", Info->AudiobufSize,audiobufsize+Info->AudiobufUsed) ); newbuf=SlibAllocBuffer(Info->AudiobufUsed+audiobufsize); if (!newbuf) return(SlibErrorMemory); memcpy(newbuf, Info->Audiobuf, Info->AudiobufUsed); SlibFreeBuffer(Info->Audiobuf); //Info->AudiobufSize+=audiobufsize; Info->Audiobuf=newbuf; } _SlibDebug(_DEBUG_, printf("Appending audio data: Info->AudiobufUsed=%d\n", Info->AudiobufUsed) ); memcpy(Info->Audiobuf+Info->AudiobufUsed, audiobuf, audiobufsize); audiobuf=Info->Audiobuf; audiobufsize+=Info->AudiobufUsed; Info->AudiobufUsed=0; } iNumBytesCompressed = audiobufsize; status = SaCompress(Info->Sah,(unsigned char *)audiobuf, &iNumBytesCompressed, NULL,&compsize); iNumBytesUnProcessed = audiobufsize - iNumBytesCompressed; //Store the Unprocessed Bytes into temp buffer if(iNumBytesUnProcessed) { //Allocate temp buff and store this audio. if (!Info->Audiobuf) { //MVP:To reduce ReAllocations and copying of memory //while checking for Unprocessed data (above),Allocate // now (normal audio buff size + Unprocessed bytes) more // memory upfront. Info->AudiobufSize=audiobufsize + iNumBytesUnProcessed; Info->Audiobuf=SlibAllocBuffer(Info->AudiobufSize); if (!Info->Audiobuf) { Info->AudiobufSize=0; return(SlibErrorMemory); } } memcpy(Info->Audiobuf, (unsigned char *)audiobuf+iNumBytesCompressed, iNumBytesUnProcessed); Info->AudiobufUsed=iNumBytesUnProcessed; } audiobufsize=iNumBytesCompressed; /* actual amount written */ } break; #endif /* G723_SUPPORT */ default: _SlibDebug(_VERBOSE_ || _WARN_, printf("SlibWriteAudio() Unsupported Format\n") ); return(SlibErrorUnsupportedFormat); } if (status==SaErrorNone && !Info->IOError) { if (Info->AudioFormat) Info->AudioTimeStamp += (audiobufsize*8000)/ (Info->AudioFormat->nSamplesPerSec*Info->AudioFormat->wBitsPerSample* Info->Channels); else Info->AudioTimeStamp += (audiobufsize*8000)/ (Info->SamplesPerSec*Info->BitsPerSample*Info->Channels); _SlibDebug(_VERBOSE_||_TIMECODE_, printf("WriteAudio(%d) Time=%ld SamplesPerSec=%d BitsPerSample=%d Channels=%d\n", audiobufsize, Info->AudioTimeStamp, Info->SamplesPerSec, Info->BitsPerSample, Info->Channels) ); } else { _SlibDebug(_WARN_, printf("SlibWriteAudio() %s\n", ScGetErrorStr(status)) ); if (status==ScErrorEndBitstream || Info->IOError) return(SlibErrorEndOfStream); return(SlibErrorWriting); } return(SlibErrorNone); } /* ** Name: slibPinReposition ** Purpose: Called when input data stream is to be repositioned. */ SlibStatus_t slibReposition(SlibInfo_t *Info, SlibPosition_t position) { SlibPin_t *pin=slibGetPin(Info, SLIB_DATA_COMPRESSED); _SlibDebug(_DEBUG_, printf("slibReposition() VideoCodecState=%d\n", Info->VideoCodecState)); if (pin) pin->Offset=position; Info->VideoPTimeCode = SLIB_TIME_NONE; Info->VideoDTimeCode = SLIB_TIME_NONE; Info->AudioPTimeCode = SLIB_TIME_NONE; Info->AudioDTimeCode = SLIB_TIME_NONE; if (Info->Fd >= 0) { _SlibDebug(_SEEK_, printf("ScFileSeek(%d, %d)\n", Info->Fd, position) ); if (ScFileSeek(Info->Fd, position)!=NoErrors) return(SlibErrorEndOfStream); return(SlibErrorNone); } else if (Info->SlibCB) { SlibMessage_t result; _SlibDebug(_VERBOSE_, printf("slibReposition() SlibCB(SLIB_MSG_REPOSITION, %d)\n", position) ); result=(*(Info->SlibCB))((SlibHandle_t)Info, SLIB_MSG_REPOSITION, (SlibCBParam1_t)position, (SlibCBParam2_t)0, (void *)Info->SlibCBUserData); if (result!=SLIB_MSG_CONTINUE) return(SlibErrorEndOfStream); return(SlibErrorNone); } return(SlibErrorForwardOnly); } /* ** Name: slibPinPrepareReposition ** Purpose: Should be called when a stream is about to be repositioned (a seek). ** This will empty any remaining buffers being used by the ** CODECs and restart them. */ void slibPinPrepareReposition(SlibInfo_t *Info, int pinid) { _SlibDebug(_DEBUG_, printf("slibPinPrepareReposition() VideoCodecState=%d\n", Info->VideoCodecState)); switch(pinid) { case SLIB_DATA_VIDEO: _SlibDebug(_VERBOSE_||_SEEK_>1, printf("slibPinPrepareReposition(Video) in\n") ); if (Info->VideoCodecState==SLIB_CODEC_STATE_BEGUN && Info->Svh && #ifdef JPEG_SUPPORT Info->Type != SLIB_TYPE_JPEG_AVI && Info->Type != SLIB_TYPE_MJPG_AVI && #endif /* JPEG_SUPPORT */ Info->Type != SLIB_TYPE_YUV_AVI) { SvDecompressEnd(Info->Svh); /* this will reset the bitstream */ Info->VideoCodecState=SLIB_CODEC_STATE_REPOSITIONING; _SlibDebug(_DEBUG_, printf("VideoCodecState=REPOSITIONING\n")); Info->IOError=FALSE; } Info->VideoPTimeCode = SLIB_TIME_NONE; Info->VideoDTimeCode = SLIB_TIME_NONE; Info->VideoTimeStamp = SLIB_TIME_NONE; Info->AvgVideoTimeDiff = 0; Info->VarVideoTimeDiff = 0; Info->VideoFramesProcessed=0; /* reset frames processed */ break; case SLIB_DATA_AUDIO: _SlibDebug(_VERBOSE_||_SEEK_>1, printf("slibPinPrepareReposition(Audio) in\n") ); if (Info->AudioCodecState==SLIB_CODEC_STATE_BEGUN && Info->Sah) { SaDecompressEnd(Info->Sah); /* this will reset the bitstream */ Info->AudioCodecState=SLIB_CODEC_STATE_REPOSITIONING; _SlibDebug(_DEBUG_, printf("AudioCodecState=REPOSITIONING\n")); Info->IOError=FALSE; } Info->AudioPTimeCode = SLIB_TIME_NONE; Info->AudioDTimeCode = SLIB_TIME_NONE; Info->AudioTimeStamp = SLIB_TIME_NONE; break; default: _SlibDebug(_VERBOSE_||_SEEK_>1, printf("slibPinPrepareReposition(%d) in\n", pinid) ); } _SlibDebug(_VERBOSE_||_SEEK_>1, printf("slibPinPrepareReposition(%d) out\n", pinid) ); } void slibPinFinishReposition(SlibInfo_t *Info, int pinid) { _SlibDebug(_DEBUG_, printf("slibPinFinishReposition() VideoCodecState=%d\n", Info->VideoCodecState)); switch(pinid) { case SLIB_DATA_VIDEO: _SlibDebug(_VERBOSE_||_SEEK_>1, printf("slibPinFinishReposition(Video) in\n") ); if (Info->VideoCodecState==SLIB_CODEC_STATE_REPOSITIONING && Info->Svh && slibGetPin(Info, SLIB_DATA_VIDEO) && #ifdef JPEG_SUPPORT Info->Type != SLIB_TYPE_JPEG_AVI && Info->Type != SLIB_TYPE_MJPG_AVI && #endif /* JPEG_SUPPORT */ Info->Type != SLIB_TYPE_YUV_AVI) slibStartVideo(Info, FALSE); break; case SLIB_DATA_AUDIO: _SlibDebug(_VERBOSE_||_SEEK_>1, printf("slibPinFinishReposition(Audio) in\n") ); if (Info->AudioCodecState==SLIB_CODEC_STATE_REPOSITIONING && Info->Sah && slibGetPin(Info, SLIB_DATA_AUDIO)) slibStartAudio(Info); break; default: _SlibDebug(_VERBOSE_||_SEEK_>1, printf("slibPinFinishReposition(%d) in\n", pinid) ); } _SlibDebug(_VERBOSE_||_SEEK_>1, printf("slibPinFinishReposition(%d) out\n", pinid) ); } SlibBoolean_t slibUpdateLengths(SlibInfo_t *Info) { if (Info->FileSize>0 && !Info->VideoLengthKnown && (SlibTimeIsValid(Info->VideoPTimeCode) || SvGetParamInt(Info->Svh, SV_PARAM_CALCTIMECODE)>=1000) ) { if (Info->VideoTimeStamp>Info->VideoLength) Info->VideoLength=Info->VideoTimeStamp; Info->VideoLengthKnown=TRUE; _SlibDebug(_SEEK_ || _TIMECODE_, printf("slibUpdateLengths() AudioLength=%ld VideoLength=%ld\n", Info->AudioLength, Info->VideoLength) ); return(TRUE); } return(FALSE); } SlibBoolean_t slibUpdatePositions(SlibInfo_t *Info, SlibBoolean_t exactonly) { SlibBoolean_t updated=FALSE; if (!exactonly) { if (SlibTimeIsValid(Info->VideoTimeStamp)) updated=TRUE; else if (SlibTimeIsValid(Info->AudioTimeStamp)) { if (slibHasVideo(Info)) Info->VideoTimeStamp=Info->AudioTimeStamp; updated=TRUE; } else if (slibHasVideo(Info)) { Info->VideoTimeStamp=slibGetNextTimeOnPin(Info, slibGetPin(Info, SLIB_DATA_VIDEO), 500*1024); if (SlibTimeIsValid(Info->VideoTimeStamp)) { Info->VideoTimeStamp-=Info->VideoPTimeBase; updated=TRUE; } } } if (!updated && (!exactonly || SlibTimeIsInValid(Info->VideoPTimeBase)) && SvGetParamInt(Info->Svh, SV_PARAM_CALCTIMECODE)>=1500 && SvGetParamInt(Info->Svh, SV_PARAM_FRAME)>24) { _SlibDebug(_TIMECODE_, printf("slibUpdatePositions() using video time\n") ); Info->VideoTimeStamp=SvGetParamInt(Info->Svh, SV_PARAM_CALCTIMECODE); updated=TRUE; } if (updated) { if (Info->VideoTimeStamp>Info->VideoLength) Info->VideoLength=Info->VideoTimeStamp; if (SlibTimeIsInValid(Info->AudioTimeStamp) && slibHasAudio(Info)) { /* need to update audio time */ Info->AudioTimeStamp=slibGetNextTimeOnPin(Info, slibGetPin(Info, SLIB_DATA_AUDIO), 100*1024); if (SlibTimeIsInValid(Info->AudioTimeStamp)) Info->AudioTimeStamp=Info->VideoTimeStamp; else Info->AudioTimeStamp-=Info->AudioPTimeBase; } if (SlibTimeIsInValid(Info->VideoPTimeCode)) Info->VideoFramesProcessed=slibTimeToFrame(Info, Info->VideoTimeStamp); } _SlibDebug(_SEEK_ || _TIMECODE_, printf("slibUpdatePositions() AudioTimeStamp=%ld VideoTimeStamp=%ld %s\n", Info->AudioTimeStamp, Info->VideoTimeStamp, updated?"updated":"") ); return(updated); } void slibAdvancePositions(SlibInfo_t *Info, qword frames) { Info->VideoFramesProcessed+=frames; if (Info->FramesPerSec) { if (Info->Mode==SLIB_MODE_COMPRESS || SlibTimeIsInValid(Info->VideoPTimeCode)) Info->VideoTimeStamp=slibFrameToTime(Info, Info->VideoFramesProcessed); else Info->VideoTimeStamp=Info->VideoPTimeCode - Info->VideoPTimeBase + slibFrameToTime(Info, Info->VideoFramesProcessed); /* Info->VideoTimeStamp+=slibFrameToTime(Info, frames); */ if (Info->VideoTimeStamp>Info->VideoLength) Info->VideoLength=Info->VideoTimeStamp; _SlibDebug(_TIMECODE_, printf("slibAdvancePositions(frames=%d) VideoTimeStamp=%ld\n", frames, Info->VideoTimeStamp) ); } } SlibStatus_t SlibSeek(SlibHandle_t handle, SlibStream_t stream, SlibSeekType_t seektype, SlibPosition_t frame) { return(SlibSeekEx(handle, stream, seektype, frame, SLIB_UNIT_FRAMES, NULL)); } SlibStatus_t SlibSeekEx(SlibHandle_t handle, SlibStream_t stream, SlibSeekType_t seektype, SlibPosition_t seekpos, SlibUnit_t seekunits, SlibSeekInfo_t *seekinfo) { SlibInfo_t *Info=(SlibInfo_t *)handle; SvStatus_t status=SlibErrorNone; SlibTime_t seektime, timediff; unsigned int tries=0; _SlibDebug(_SEEK_, printf("SlibSeekEx(stream=%d,seektype=%d,pos=%ld,units=%d)\n", stream,seektype,seekpos,seekunits) ); if (!handle) return(SlibErrorBadHandle); if (Info->Mode!=SLIB_MODE_DECOMPRESS && seektype!=SLIB_SEEK_RESET) return(SlibErrorBadMode); if (SlibSeekTypeUsesPosition(seektype)) { switch (seekunits) { case SLIB_UNIT_FRAMES: /* frames */ /* we need to convert the frame to the time */ if (Info->FramesPerSec) seektime=slibFrameToTime(Info, seekpos); else seektime=SLIB_TIME_NONE; break; case SLIB_UNIT_MS: /* milliseconds */ seektime=(seekpos<0) ? 0 : seekpos; break; case SLIB_UNIT_PERCENT100: /* 100th of a percent */ if (seekpos<0 || seekpos>10000) return(SlibErrorBadPosition); if (Info->VideoStreams==0 || stream==SLIB_STREAM_MAINAUDIO) seektime=(seekpos * Info->AudioLength)/(SlibPosition_t)10000; else seektime=(seekpos * Info->VideoLength)/(SlibPosition_t)10000; break; default: return(SlibErrorBadUnit); } /* see if the new position is past the end of the file */ if (Info->VideoLengthKnown && seektime>Info->VideoLength) return(SlibErrorBadPosition); } else seektime=(seekpos<0) ? 0 : seekpos; if (seekinfo) seekinfo->FramesSkipped=0; switch(seektype) { case SLIB_SEEK_AHEAD: _SlibDebug(_VERBOSE_||_SEEK_, printf("SlibSeekEx(stream=%d,AHEAD,time=%d) VideoTimeStamp=%ld\n", stream,seektime,Info->VideoTimeStamp) ); if (seektime<=0) return(SlibErrorNone); if (stream==SLIB_STREAM_MAINAUDIO || Info->VideoStreams<=0) seektime+=Info->AudioTimeStamp; else seektime+=Info->VideoTimeStamp; return(SlibSeekEx(handle, stream, SLIB_SEEK_NEXT_NEAR, seektime, SLIB_UNIT_MS, seekinfo)); case SLIB_SEEK_NEXT_NEAR: _SlibDebug(_VERBOSE_||_SEEK_, printf("SlibSeekEx(stream=%d,NEXT_NEAR,time=%d) VideoTimeStamp=%ld\n", stream,seektime,Info->VideoTimeStamp) ); status=slibStartVideo(Info, FALSE); if (status==SlibErrorNone) { qword framesskipped=0; SlibBoolean_t atkey=FALSE; /* key's must be decoded */ if (Info->Svh) /* update key spacing */ { Info->KeySpacing=(int)SvGetParamInt(Info->Svh, SV_PARAM_KEYSPACING); Info->SubKeySpacing=(int)SvGetParamInt(Info->Svh, SV_PARAM_SUBKEYSPACING); } timediff=seektime-Info->VideoTimeStamp; while (status==SlibErrorNone && (timediff>500 || timediff>(Info->VideoFrameDuration*Info->KeySpacing*6)/1000)) { status=SlibSeekEx(handle, stream, SLIB_SEEK_NEXT_KEY, 0, SLIB_UNIT_NONE, seekinfo); if (seekinfo) framesskipped+=seekinfo->FramesSkipped; timediff=seektime-Info->VideoTimeStamp; atkey=TRUE; } if (!atkey && status==SlibErrorNone && (timediff>150 || timediff>(Info->VideoFrameDuration*Info->SubKeySpacing*6)/1000)) { if (SvGetParamInt(Info->Svh, SV_PARAM_FRAMETYPE)!=FRAME_TYPE_I && SvGetParamInt(Info->Svh, SV_PARAM_FRAMETYPE)!=FRAME_TYPE_P) { /* we're seeking past more than one frame */ status=SlibSeekEx(handle, stream, SLIB_SEEK_NEXT_SUBKEY, 0, SLIB_UNIT_NONE, seekinfo); if (seekinfo) framesskipped+=seekinfo->FramesSkipped; timediff=seektime-Info->VideoTimeStamp; } atkey=TRUE; } while (!atkey && status==SlibErrorNone && timediff>Info->VideoFrameDuration/100) { if (SvGetParamInt(Info->Svh, SV_PARAM_FRAMETYPE)==FRAME_TYPE_B || SvGetParamInt(Info->Svh, SV_PARAM_FRAMETYPE)==FRAME_TYPE_NONE) { /* we can skip B frames without decompressing them */ status=SlibSeekEx(handle, stream, SLIB_SEEK_NEXT, 0, SLIB_UNIT_NONE, seekinfo); if (seekinfo) framesskipped+=seekinfo->FramesSkipped; timediff=seektime-Info->VideoTimeStamp; } else atkey=TRUE; } if (seekinfo) seekinfo->FramesSkipped=framesskipped; } return(status); case SLIB_SEEK_NEXT: _SlibDebug(_VERBOSE_||_SEEK_, printf("SlibSeekEx(stream=%d,NEXT,time=%d) VideoTimeStamp=%ld\n", stream,seektime,Info->VideoTimeStamp) ); if ((status=slibStartVideo(Info, FALSE))!=SlibErrorNone) return(status); #ifdef MPEG_SUPPORT if (Info->VideoCodecState==SLIB_CODEC_STATE_BEGUN && Info->Svh && SlibTypeIsMPEGVideo(Info->Type)) { SvPictureInfo_t mpegPictureInfo; unsigned char *videobuf; /* cannot skip I or B frames without decompressing them */ if (SvGetParamInt(Info->Svh, SV_PARAM_FRAMETYPE)==FRAME_TYPE_I || SvGetParamInt(Info->Svh, SV_PARAM_FRAMETYPE)==FRAME_TYPE_P) { _SlibDebug(_DEBUG_||_SEEK_, printf("SvDecompressMPEG()\n") ); status = SvDecompressMPEG(Info->Svh, Info->Multibuf, Info->MultibufSize, &videobuf); _SlibDebug(_WARN_ && status!=SvErrorNone, printf("SvDecompressMPEG() %s\n", ScGetErrorStr(status)) ); } else { mpegPictureInfo.Type = SV_I_PICTURE|SV_P_PICTURE|SV_B_PICTURE; _SlibDebug(_VERBOSE_||_SEEK_>1, printf("SvFindNextPicture(I|P|B)\n") ); status = SvFindNextPicture(Info->Svh, &mpegPictureInfo); _SlibDebug(_WARN_ && status!=SvErrorNone, printf("SvFindNextPicture() %s\n", ScGetErrorStr(status)) ); } if (status==NoErrors) { slibAdvancePositions(Info, 1); if (Info->stats && Info->stats->Record) Info->stats->FramesSkipped++; if (seekinfo) seekinfo->FramesSkipped++; } else if (status==ScErrorEndBitstream) { if (Info->FileSize>0 && !Info->VideoLengthKnown) slibUpdateLengths(Info); return(SlibErrorEndOfStream); } return(SlibErrorNone); } #endif /* MPEG_SUPPORT */ return(SlibErrorReading); case SLIB_SEEK_EXACT: _SlibDebug(_VERBOSE_||_SEEK_, printf("SlibSeekEx(stream=%d,EXACT,time=%d) VideoTimeStamp=%ld\n", stream,seektime,Info->VideoTimeStamp) ); if (Info->FileSize<=0) return(SlibErrorFileSize); if (seektime==0 || Info->VideoStreams<=0) return(SlibSeekEx(handle, stream, SLIB_SEEK_KEY, 0, SLIB_UNIT_MS, seekinfo)); timediff=seektime-Info->VideoTimeStamp; if ((stream==SLIB_STREAM_MAINVIDEO || Info->AudioStreams==0) && (timediff>=-20 && timediff<=20)) /* we're already near the frame */ return(SlibErrorNone); if (Info->Svh) /* update key spacing */ Info->KeySpacing=(int)SvGetParamInt(Info->Svh, SV_PARAM_KEYSPACING); if (timediff>(Info->KeySpacing*Info->VideoFrameDuration)/100 || timediff<0 || (stream!=SLIB_STREAM_MAINVIDEO && Info->AudioStreams>0)) { if (Info->KeySpacing>1) { const SlibTime_t keytime= (Info->VideoFrameDuration*Info->KeySpacing)/100; if (seektime>=0 && seektimeVideoTimeStamp; } #if 0 if (SvGetParamFloat(Info->Svh, SV_PARAM_FPS)>0) Info->FramesPerSec=SvGetParamFloat(Info->Svh, SV_PARAM_FPS); #endif if (status==SlibErrorNone && timediff>=Info->VideoFrameDuration/100) { dword framesskipped=0; do { status=SlibSeekEx(handle, stream, SLIB_SEEK_NEXT, 0, SLIB_UNIT_NONE, seekinfo); framesskipped++; timediff=seektime-Info->VideoTimeStamp; } while (timediff>0 && status==SlibErrorNone); if (seekinfo) seekinfo->FramesSkipped+=framesskipped; /* if we skipped some frames, skip some audio too */ if (framesskipped>5 && stream==SLIB_STREAM_ALL && Info->AudioStreams>0) { slibPinPrepareReposition(Info, SLIB_DATA_AUDIO); slibSkipAudio(Info, stream, (Info->VideoFrameDuration* framesskipped)/100); slibPinFinishReposition(Info, SLIB_DATA_AUDIO); } } return(status); case SLIB_SEEK_NEXT_KEY: _SlibDebug(_VERBOSE_||_SEEK_, printf("SlibSeekEx(stream=%d,NEXT_KEY,time=%d) VideoTimeStamp=%ld\n", stream,seektime,Info->VideoTimeStamp) ); if ((status=slibStartVideo(Info, FALSE))!=SlibErrorNone) return(status); #ifdef MPEG_SUPPORT if (Info->Svh && SlibTypeIsMPEGVideo(Info->Type)) { SvPictureInfo_t mpegPictureInfo; SlibTime_t vtime=Info->VideoTimeStamp; do { mpegPictureInfo.Type = SV_I_PICTURE; status = SvFindNextPicture(Info->Svh, &mpegPictureInfo); if (status==NoErrors && mpegPictureInfo.Type==SV_I_PICTURE) { if (Info->stats && Info->stats->Record) Info->stats->FramesSkipped+=mpegPictureInfo.TemporalRef; if (vtime==Info->VideoTimeStamp) /* timecode didn't update time */ slibAdvancePositions(Info, mpegPictureInfo.TemporalRef); vtime=Info->VideoTimeStamp; if (seekinfo) { seekinfo->FramesSkipped+=mpegPictureInfo.TemporalRef; seekinfo->VideoTimeStamp=Info->VideoTimeStamp; seekinfo->AudioTimeStamp=Info->AudioTimeStamp; } return(SlibErrorNone); } } while (status==NoErrors); if (seekinfo) { seekinfo->VideoTimeStamp=Info->VideoTimeStamp; seekinfo->AudioTimeStamp=Info->AudioTimeStamp; } if (status==ScErrorEndBitstream) { if (Info->FileSize>0 && !Info->VideoLengthKnown) slibUpdateLengths(Info); return(SlibErrorEndOfStream); } } _SlibDebug(_WARN_, printf("SvFindNextPicture() %s\n", ScGetErrorStr(status)) ); #endif /* MPEG_SUPPORT */ /* do an absolute seek */ status=SlibSeekEx(handle, stream, SLIB_SEEK_KEY, (Info->VideoStreams<=0?Info->AudioTimeStamp :Info->VideoTimeStamp)+1000, SLIB_UNIT_MS, seekinfo); return(status); case SLIB_SEEK_NEXT_SUBKEY: _SlibDebug(_VERBOSE_||_SEEK_, printf("SlibSeekEx(stream=%d,NEXT_SUBKEY,time=%d) VideoTime=%ld\n", stream,seektime,Info->VideoTimeStamp) ); if ((status=slibStartVideo(Info, FALSE))!=SlibErrorNone) return(status); #ifdef MPEG_SUPPORT if (Info->Svh && SlibTypeIsMPEGVideo(Info->Type)) { SvPictureInfo_t mpegPictureInfo; unsigned char *videobuf; SlibTime_t vtime=Info->VideoTimeStamp; /* cannot skip I or B frames without decompressing them */ if (SvGetParamInt(Info->Svh, SV_PARAM_FRAMETYPE)==FRAME_TYPE_I || SvGetParamInt(Info->Svh, SV_PARAM_FRAMETYPE)==FRAME_TYPE_P) { _SlibDebug(_DEBUG_||_SEEK_, printf("SvDecompressMPEG()\n") ); status = SvDecompressMPEG(Info->Svh, Info->Multibuf, Info->MultibufSize, &videobuf); if (vtime==Info->VideoTimeStamp /* timecode didn't update time */ && status==SvErrorNone) slibAdvancePositions(Info, 1); vtime=Info->VideoTimeStamp; if (seekinfo) seekinfo->FramesSkipped+=mpegPictureInfo.TemporalRef; _SlibDebug(_WARN_ && status!=SvErrorNone, printf("SvDecompressMPEG() %s\n", ScGetErrorStr(status)) ); } do { mpegPictureInfo.Type = SV_I_PICTURE|SV_P_PICTURE; status = SvFindNextPicture(Info->Svh, &mpegPictureInfo); if (Info->stats && Info->stats->Record) Info->stats->FramesSkipped+=mpegPictureInfo.TemporalRef; if (vtime==Info->VideoTimeStamp) /* timecode didn't update time */ slibAdvancePositions(Info, mpegPictureInfo.TemporalRef); vtime=Info->VideoTimeStamp; if (seekinfo) seekinfo->FramesSkipped+=mpegPictureInfo.TemporalRef; if (mpegPictureInfo.Type == SV_I_PICTURE || mpegPictureInfo.Type == SV_P_PICTURE) { /* found a subkey frame */ if (seekinfo) { seekinfo->VideoTimeStamp=Info->VideoTimeStamp; seekinfo->AudioTimeStamp=Info->AudioTimeStamp; } return(SlibErrorNone); } } while(status==NoErrors); if (status==ScErrorEndBitstream) { if (Info->FileSize>0 && !Info->VideoLengthKnown) slibUpdateLengths(Info); if (seekinfo) { seekinfo->VideoTimeStamp=Info->VideoTimeStamp; seekinfo->AudioTimeStamp=Info->AudioTimeStamp; } return(SlibErrorEndOfStream); } } _SlibDebug(_WARN_, printf("SvFindNextPicture() %s\n", ScGetErrorStr(status)) ); #endif /* MPEG_SUPPORT */ /* do an absolute seek */ status=SlibSeekEx(handle, stream, SLIB_SEEK_KEY, (Info->VideoStreams<=0?Info->AudioTimeStamp :Info->VideoTimeStamp)+500, SLIB_UNIT_MS, seekinfo); return(status); case SLIB_SEEK_KEY: _SlibDebug(_VERBOSE_||_SEEK_, printf("SlibSeekEx(stream=%d,KEY,time=%d) VideoTimeStamp=%ld\n", stream,seektime,Info->VideoTimeStamp) ); if (!Info->HeaderProcessed) { /* At very start of file we must Start the codecs since they * may need crucial header info */ status=slibStartVideo(Info, FALSE); if (status!=SlibErrorNone) return(status); } if (Info->FileSize<=0) return(SlibErrorFileSize); if (seekpos!=0 && seekunits!=SLIB_UNIT_PERCENT100 && (stream==SLIB_STREAM_MAINVIDEO || Info->AudioStreams==0) && SlibTimeIsValid(Info->VideoTimeStamp)) { /* see if we're already near the frame */ timediff=seektime-Info->VideoTimeStamp; if (timediff>=-33 && timediff<=33) return(SlibErrorNone); } if ((seekunits==SLIB_UNIT_PERCENT100 && seekpos<=50) || seektime<=slibTimeToFrame(Info, 6)) { /* close to beginning */ if (seektime<=(Info->VideoFrameDuration*2)/100 && stream==SLIB_STREAM_MAINVIDEO) /* already close enough */ return(SlibErrorNone); seek_to_beginning: if (stream==SLIB_STREAM_ALL || stream==SLIB_STREAM_MAINVIDEO) slibPinPrepareReposition(Info, SLIB_DATA_VIDEO); if (stream==SLIB_STREAM_ALL || stream==SLIB_STREAM_MAINAUDIO) slibPinPrepareReposition(Info, SLIB_DATA_AUDIO); if (stream==SLIB_STREAM_ALL || stream==SLIB_STREAM_MAINVIDEO) slibEmptyPin(Info, SLIB_DATA_VIDEO); if (stream==SLIB_STREAM_ALL || stream==SLIB_STREAM_MAINAUDIO) slibEmptyPin(Info, SLIB_DATA_AUDIO); slibEmptyPin(Info, SLIB_DATA_COMPRESSED); if ((status=slibReposition(Info, 0))!=SlibErrorNone) return(status); Info->IOError=FALSE; Info->VideoTimeStamp = slibHasVideo(Info) ? 0 : SLIB_TIME_NONE; Info->AudioTimeStamp = slibHasAudio(Info) ? 0 : SLIB_TIME_NONE; return(SlibErrorNone); } else { qword skippedframes=0; unsigned qword filepos; SlibTime_t vtime; const qword length=(Info->VideoStreams<=0) ? Info->AudioLength : Info->VideoLength; if (seekunits==SLIB_UNIT_PERCENT100) { unsigned qword bytes_between_keys=Info->TotalBitRate/(8*2); filepos = (seekpos*Info->FileSize)/10000; /* seek a little before the desired point */ if (bytes_between_keys>filepos) goto seek_to_beginning; else filepos-=bytes_between_keys; } else if (length==0) goto seek_to_beginning; else if (Info->FileSize<0x100000000)/* be careful of mul overflow */ filepos = (seektime*Info->FileSize)/length; else filepos = ((seektime/100)*Info->FileSize)/(length/100); seek_to_key: if (stream==SLIB_STREAM_ALL || stream==SLIB_STREAM_MAINVIDEO) slibPinPrepareReposition(Info, SLIB_DATA_VIDEO); if (stream==SLIB_STREAM_ALL || stream==SLIB_STREAM_MAINAUDIO) slibPinPrepareReposition(Info, SLIB_DATA_AUDIO); if (stream==SLIB_STREAM_ALL || stream==SLIB_STREAM_MAINVIDEO) slibEmptyPin(Info, SLIB_DATA_VIDEO); if (stream==SLIB_STREAM_ALL || stream==SLIB_STREAM_MAINAUDIO) slibEmptyPin(Info, SLIB_DATA_AUDIO); slibEmptyPin(Info, SLIB_DATA_COMPRESSED); if ((status=slibReposition(Info, filepos))!=SlibErrorNone) return(status); Info->IOError=FALSE; if (stream==SLIB_STREAM_ALL || stream==SLIB_STREAM_MAINVIDEO) slibPinFinishReposition(Info, SLIB_DATA_VIDEO); if (stream==SLIB_STREAM_ALL || stream==SLIB_STREAM_MAINAUDIO) slibPinFinishReposition(Info, SLIB_DATA_AUDIO); vtime=Info->VideoTimeStamp; #ifdef MPEG_SUPPORT if (Info->Svh && SlibTypeIsMPEGVideo(Info->Type)) { SvPictureInfo_t mpegPictureInfo; if ((status=slibStartVideo(Info, FALSE))!=SlibErrorNone) return(status); mpegPictureInfo.Type = SV_I_PICTURE; status = SvFindNextPicture(Info->Svh, &mpegPictureInfo); _SlibDebug(_WARN_ && status!=NoErrors, printf("SvFindNextPicture() %s\n", ScGetErrorStr(status)) ); if (status!=NoErrors) return(SlibErrorEndOfStream); skippedframes=mpegPictureInfo.TemporalRef; } #endif /* MPEG_SUPPORT */ if (seekunits==SLIB_UNIT_PERCENT100) { /* See if we seeked to far ahead */ SlibPosition_t posdiff= SlibGetParamInt(Info, SLIB_STREAM_ALL, SLIB_PARAM_PERCENT100) -seekpos; if (filepos>0 && posdiff>0 && tries<2) { tries++; /* we're ahead by one percent or more */ /* move filepos back in the proportion that we're off by */ filepos-=(posdiff*Info->FileSize)/8000; if (filepos<0) goto seek_to_beginning; goto seek_to_key; } } if (slibUpdatePositions(Info, FALSE)) /* timecoded */ { /* timecoded */ /* * See if we seeked to far ahead * Note: if times are way off then we should ignore them. */ if (seekunits==SLIB_UNIT_PERCENT100) /* ignore times */ timediff=0; else { timediff=seektime-Info->VideoTimeStamp; if (timediff>-5000 && timediff<-100 && tries<3) { /* move filepos back in the proportion that we're off by */ filepos=(filepos*seektime)/Info->VideoTimeStamp; if (filepos<0) filepos=0; tries++; goto seek_to_key; } } #ifdef MPEG_SUPPORT if (Info->Svh && SlibTypeIsMPEGVideo(Info->Type)) { SvPictureInfo_t mpegPictureInfo; mpegPictureInfo.Type = SV_I_PICTURE; while (timediff>Info->VideoFrameDuration/100 &&status==NoErrors) { _SlibDebug(_SEEK_>1, printf("SlibSeekEx(KEY, %d) Find next I frame (%d/%d)\n", seektime, SvGetParamInt(Info->Svh, SV_PARAM_CALCTIMECODE), Info->VideoTimeStamp) ); status = SvFindNextPicture(Info->Svh, &mpegPictureInfo); _SlibDebug(_WARN_ && status!=NoErrors, printf("SvFindNextPicture() %s\n", ScGetErrorStr(status)) ); skippedframes+=mpegPictureInfo.TemporalRef; if (vtime==Info->VideoTimeStamp) /* timecode didn't update time */ slibAdvancePositions(Info, mpegPictureInfo.TemporalRef); vtime=Info->VideoTimeStamp; timediff=seektime-Info->VideoTimeStamp; } } #endif /* MPEG_SUPPORT */ } else { _SlibDebug(_SEEK_, printf("SlibSeekEx(KEY, %d) no timecode\n", seektime) ); if (slibHasVideo(Info)) { Info->VideoTimeStamp=seektime; Info->VideoFramesProcessed=slibTimeToFrame(Info, seektime); } if (slibHasAudio(Info)) Info->AudioTimeStamp=seektime; slibAdvancePositions(Info, skippedframes); timediff=seektime-Info->VideoTimeStamp; } if (Info->stats && Info->stats->Record) Info->stats->FramesSkipped+=skippedframes; if (seekinfo) seekinfo->FramesSkipped=skippedframes; #if 0 if (Info->Svh) Info->FramesPerSec=SvGetParamFloat(Info->Svh, SV_PARAM_FPS); #endif /* if we skipped some frames, skip some audio too */ if (skippedframes>5 && stream==SLIB_STREAM_ALL && slibHasAudio(Info)) { slibPinPrepareReposition(Info, SLIB_DATA_AUDIO); slibSkipAudio(Info, stream, (Info->VideoFrameDuration* skippedframes)/100); slibPinFinishReposition(Info, SLIB_DATA_AUDIO); if (SlibTimeIsInValid(Info->AudioTimeStamp)) { Info->AudioTimeStamp=slibGetNextTimeOnPin(Info, slibGetPin(Info, SLIB_DATA_AUDIO), 100*1024); if (SlibTimeIsInValid(Info->AudioTimeStamp)) Info->AudioTimeStamp=Info->VideoTimeStamp; else Info->AudioTimeStamp-=Info->AudioPTimeBase; } } if (status==ScErrorEndBitstream) { if (Info->FileSize>0 && !Info->VideoLengthKnown) slibUpdateLengths(Info); return(SlibErrorEndOfStream); } else if (status!=NoErrors) return(SlibErrorReading); return(SlibErrorNone); } break; case SLIB_SEEK_RESET: _SlibDebug(_VERBOSE_||_SEEK_, printf("SlibSeekEx(stream=%d,RESET,time=%d) VideoTimeStamp=%ld\n", stream,seektime,Info->VideoTimeStamp) ); if (stream==SLIB_STREAM_ALL || stream==SLIB_STREAM_MAINVIDEO) slibPinPrepareReposition(Info, SLIB_DATA_VIDEO); if (stream==SLIB_STREAM_ALL || stream==SLIB_STREAM_MAINAUDIO) slibPinPrepareReposition(Info, SLIB_DATA_AUDIO); if (stream==SLIB_STREAM_ALL || stream==SLIB_STREAM_MAINVIDEO) slibEmptyPin(Info, SLIB_DATA_VIDEO); if (stream==SLIB_STREAM_ALL || stream==SLIB_STREAM_MAINAUDIO) slibEmptyPin(Info, SLIB_DATA_AUDIO); if (stream==SLIB_STREAM_MAINAUDIO) slibPinFinishReposition(Info, SLIB_DATA_AUDIO); if (stream==SLIB_STREAM_ALL) { slibEmptyPin(Info, SLIB_DATA_COMPRESSED); Info->BytesProcessed = 0; } Info->HeaderProcessed = FALSE; return(SlibErrorNone); case SLIB_SEEK_RESYNC: seekpos=SlibGetParamInt(Info, SLIB_STREAM_ALL, SLIB_PARAM_PERCENT100); if (seekpos<0 || (SlibTimeIsValid(Info->VideoTimeStamp) && Info->VideoTimeStampVideoTimeStamp) ); } return(SlibErrorForwardOnly); } SlibList_t *SlibQueryList(SlibQueryType_t qtype) { switch(qtype) { case SLIB_QUERY_TYPES: return(_listTypes); case SLIB_QUERY_COMP_TYPES: return(_listCompressTypes); case SLIB_QUERY_DECOMP_TYPES: return(_listDecompressTypes); case SLIB_QUERY_ERRORS: return(_listErrors); default: return(NULL); } } char *SlibQueryForDesc(SlibQueryType_t qtype, int enumval) { SlibList_t *entry=SlibQueryList(qtype); if (entry) entry=SlibFindEnumEntry(entry, enumval); if (entry) return(entry->Desc); else return(NULL); } int SlibQueryForEnum(SlibQueryType_t qtype, char *name) { SlibList_t *list=SlibQueryList(qtype); if (!list) return(-1); while (list->Name) { if (strcmp(list->Name, name)==0) return(list->Enum); list++; } return(-1); } SlibBoolean_t SlibIsEnd(SlibHandle_t handle, SlibStream_t stream) { SlibInfo_t *Info=(SlibInfo_t *)handle; SlibBoolean_t isend=FALSE; if (!handle) isend=TRUE; else if (stream==SLIB_STREAM_MAINAUDIO) { isend=SlibPeekBuffer(Info, SLIB_DATA_AUDIO, NULL, NULL)==NULL?TRUE:FALSE; if (isend && Info->Sah) /* see if the audio codec still has data */ { ScBitstream_t *bs=SaGetDataSource(Info->Sah); if (bs && !bs->EOI) isend=FALSE; } } else if (stream==SLIB_STREAM_MAINVIDEO) { isend=SlibPeekBuffer(Info, SLIB_DATA_VIDEO, NULL, NULL)==NULL?TRUE:FALSE; if (isend && Info->Svh) /* see if the video codec still has data */ { ScBitstream_t *bs=SvGetDataSource(Info->Svh); if (bs && !bs->EOI) isend=FALSE; } } else if (SlibPeekBuffer(Info, SLIB_DATA_AUDIO, NULL, NULL)==NULL && SlibPeekBuffer(Info, SLIB_DATA_VIDEO, NULL, NULL)==NULL && SlibPeekBuffer(Info, SLIB_DATA_COMPRESSED, NULL, NULL)==NULL) { ScBitstream_t *bs; isend=TRUE; if (Info->Svh) /* see if the video codec still has data */ { bs=SvGetDataSource(Info->Svh); if (bs && !bs->EOI) isend=FALSE; } if (isend && Info->Sah) /* see if the audio codec still has data */ { bs=SaGetDataSource(Info->Sah); if (bs && !bs->EOI) isend=FALSE; } } _SlibDebug(_VERBOSE_, printf("SlibIsEnd() %s\n",isend?"TRUE":"FALSE")); return(isend); } SlibStatus_t SlibClose(SlibHandle_t handle) { SlibInfo_t *Info=(SlibInfo_t *)handle; _SlibDebug(_DEBUG_, printf("SlibClose\n") ); if (!handle) return(SlibErrorBadHandle); /* close video codec */ if (Info->Svh) { if (Info->VideoCodecState==SLIB_CODEC_STATE_BEGUN) { _SlibDebug(_DEBUG_, printf("SvDecompress/CompressEnd()\n") ); if (Info->Mode==SLIB_MODE_DECOMPRESS) SvDecompressEnd(Info->Svh); else if (Info->Mode==SLIB_MODE_COMPRESS) SvCompressEnd(Info->Svh); Info->VideoCodecState=SLIB_CODEC_STATE_INITED; } _SlibDebug(_DEBUG_, printf("SvCloseCodec()\n") ); SvCloseCodec(Info->Svh); } Info->VideoCodecState=SLIB_CODEC_STATE_NONE; /* close audio codec */ if (Info->Sah) { if (Info->AudioCodecState==SLIB_CODEC_STATE_BEGUN) { if (Info->Mode==SLIB_MODE_DECOMPRESS) SaDecompressEnd(Info->Sah); else if (Info->Mode==SLIB_MODE_COMPRESS) SaCompressEnd(Info->Sah); Info->AudioCodecState=SLIB_CODEC_STATE_INITED; } _SlibDebug(_DEBUG_, printf("SaCloseCodec()\n") ); SaCloseCodec(Info->Sah); } Info->AudioCodecState=SLIB_CODEC_STATE_NONE; if (Info->Mode==SLIB_MODE_COMPRESS && Info->HeaderProcessed) slibCommitBuffers(Info, TRUE); /* close format converter */ if (Info->Sch) { SconClose(Info->Sch); Info->Sch=NULL; } /* close data sources */ if (Info->Fd>=0) { _SlibDebug(_DEBUG_, printf("ScFileClose(%d)\n",Info->Fd) ); ScFileClose(Info->Fd); Info->Fd=-1; } slibRemovePins(Info); /* slibDumpMemory(); */ if (Info->SlibCB) { SlibMessage_t result; _SlibDebug(_VERBOSE_, printf("SlibClose() SlibCB(SLIB_MSG_CLOSE)\n") ); result=(*(Info->SlibCB))((SlibHandle_t)Info, SLIB_MSG_CLOSE, (SlibCBParam1_t)0, (SlibCBParam2_t)0, (void *)Info->SlibCBUserData); Info->SlibCB=NULL; } /* free memory */ if (Info->stats) ScFree(Info->stats); if (Info->CompVideoFormat) ScFree(Info->CompVideoFormat); if (Info->CodecVideoFormat) ScFree(Info->CodecVideoFormat); if (Info->VideoFormat) ScFree(Info->VideoFormat); if (Info->AudioFormat) ScFree(Info->AudioFormat); if (Info->CompAudioFormat) ScFree(Info->CompAudioFormat); if (Info->Imagebuf) SlibFreeBuffer(Info->Imagebuf); if (Info->CodecImagebuf) SlibFreeBuffer(Info->CodecImagebuf); if (Info->IntImagebuf) SlibFreeBuffer(Info->IntImagebuf); if (Info->Audiobuf) SlibFreeBuffer(Info->Audiobuf); if (Info->Multibuf) /* free all outstanding allocations on Multibuf */ while (SlibFreeBuffer(Info->Multibuf)==SlibErrorNone); ScFree(Info); _SlibDebug(_WARN_ && SlibMemUsed()>0, printf("SlibClose() mem used=%d\n", SlibMemUsed()) ); return(SlibErrorNone); } /* SlibStatus_t SlibGetInfo(SlibHandle_t handle, SlibInfo_t *info) { if (!handle) return(SlibErrorBadHandle); if (!info) return(SlibErrorBadArgument); memcpy(info, handle, sizeof(SlibInfo_t)); return(SlibErrorNone); } */